5d86e5967d22e80f12f4f62c063274cce4daa35e
[roojs1] / roojs-debug.js
1 /*
2  * Based on:
3  * Ext JS Library 1.1.1
4  * Copyright(c) 2006-2007, Ext JS, LLC.
5  *
6  * Originally Released Under LGPL - original licence link has changed is not relivant.
7  *
8  * Fork - LGPL
9  * <script type="text/javascript">
10  */
11  
12
13
14
15
16 // for old browsers
17 window["undefined"] = window["undefined"];
18
19 /**
20  * @class Roo
21  * Roo core utilities and functions.
22  * @singleton
23  */
24 var Roo = {}; 
25 /**
26  * Copies all the properties of config to obj.
27  * @param {Object} obj The receiver of the properties
28  * @param {Object} config The source of the properties
29  * @param {Object} defaults A different object that will also be applied for default values
30  * @return {Object} returns obj
31  * @member Roo apply
32  */
33
34  
35 Roo.apply = function(o, c, defaults){
36     if(defaults){
37         // no "this" reference for friendly out of scope calls
38         Roo.apply(o, defaults);
39     }
40     if(o && c && typeof c == 'object'){
41         for(var p in c){
42             o[p] = c[p];
43         }
44     }
45     return o;
46 };
47
48
49 (function(){
50     var idSeed = 0;
51     var ua = navigator.userAgent.toLowerCase();
52
53     var isStrict = document.compatMode == "CSS1Compat",
54         isOpera = ua.indexOf("opera") > -1,
55         isSafari = (/webkit|khtml/).test(ua),
56         isFirefox = ua.indexOf("firefox") > -1,
57         isIE = ua.indexOf("msie") > -1,
58         isIE7 = ua.indexOf("msie 7") > -1,
59         isIE11 = /trident.*rv\:11\./.test(ua),
60         isEdge = ua.indexOf("edge") > -1,
61         isGecko = !isSafari && ua.indexOf("gecko") > -1,
62         isBorderBox = isIE && !isStrict,
63         isWindows = (ua.indexOf("windows") != -1 || ua.indexOf("win32") != -1),
64         isMac = (ua.indexOf("macintosh") != -1 || ua.indexOf("mac os x") != -1),
65         isLinux = (ua.indexOf("linux") != -1),
66         isSecure = window.location.href.toLowerCase().indexOf("https") === 0,
67         isIOS = /iphone|ipad/.test(ua),
68         isAndroid = /android/.test(ua),
69         isTouch =  (function() {
70             try {
71                 if (ua.indexOf('chrome') != -1 && ua.indexOf('android') == -1) {
72                     window.addEventListener('touchstart', function __set_has_touch__ () {
73                         Roo.isTouch = true;
74                         window.removeEventListener('touchstart', __set_has_touch__);
75                     });
76                     return false; // no touch on chrome!?
77                 }
78                 document.createEvent("TouchEvent");  
79                 return true;  
80             } catch (e) {  
81                 return false;  
82             } 
83             
84         })();
85     // remove css image flicker
86         if(isIE && !isIE7){
87         try{
88             document.execCommand("BackgroundImageCache", false, true);
89         }catch(e){}
90     }
91     
92     Roo.apply(Roo, {
93         /**
94          * True if the browser is in strict mode
95          * @type Boolean
96          */
97         isStrict : isStrict,
98         /**
99          * True if the page is running over SSL
100          * @type Boolean
101          */
102         isSecure : isSecure,
103         /**
104          * True when the document is fully initialized and ready for action
105          * @type Boolean
106          */
107         isReady : false,
108         /**
109          * Turn on debugging output (currently only the factory uses this)
110          * @type Boolean
111          */
112         
113         debug: false,
114
115         /**
116          * True to automatically uncache orphaned Roo.Elements periodically (defaults to true)
117          * @type Boolean
118          */
119         enableGarbageCollector : true,
120
121         /**
122          * True to automatically purge event listeners after uncaching an element (defaults to false).
123          * Note: this only happens if enableGarbageCollector is true.
124          * @type Boolean
125          */
126         enableListenerCollection:false,
127
128         /**
129          * URL to a blank file used by Roo when in secure mode for iframe src and onReady src to prevent
130          * the IE insecure content warning (defaults to javascript:false).
131          * @type String
132          */
133         SSL_SECURE_URL : "javascript:false",
134
135         /**
136          * URL to a 1x1 transparent gif image used by Roo to create inline icons with CSS background images. (Defaults to
137          * "http://Roojs.com/s.gif" and you should change this to a URL on your server).
138          * @type String
139          */
140         BLANK_IMAGE_URL : "http:/"+"/localhost/s.gif",
141
142         emptyFn : function(){},
143         
144         /**
145          * Copies all the properties of config to obj if they don't already exist.
146          * @param {Object} obj The receiver of the properties
147          * @param {Object} config The source of the properties
148          * @return {Object} returns obj
149          */
150         applyIf : function(o, c){
151             if(o && c){
152                 for(var p in c){
153                     if(typeof o[p] == "undefined"){ o[p] = c[p]; }
154                 }
155             }
156             return o;
157         },
158
159         /**
160          * Applies event listeners to elements by selectors when the document is ready.
161          * The event name is specified with an @ suffix.
162 <pre><code>
163 Roo.addBehaviors({
164    // add a listener for click on all anchors in element with id foo
165    '#foo a@click' : function(e, t){
166        // do something
167    },
168
169    // add the same listener to multiple selectors (separated by comma BEFORE the @)
170    '#foo a, #bar span.some-class@mouseover' : function(){
171        // do something
172    }
173 });
174 </code></pre>
175          * @param {Object} obj The list of behaviors to apply
176          */
177         addBehaviors : function(o){
178             if(!Roo.isReady){
179                 Roo.onReady(function(){
180                     Roo.addBehaviors(o);
181                 });
182                 return;
183             }
184             var cache = {}; // simple cache for applying multiple behaviors to same selector does query multiple times
185             for(var b in o){
186                 var parts = b.split('@');
187                 if(parts[1]){ // for Object prototype breakers
188                     var s = parts[0];
189                     if(!cache[s]){
190                         cache[s] = Roo.select(s);
191                     }
192                     cache[s].on(parts[1], o[b]);
193                 }
194             }
195             cache = null;
196         },
197
198         /**
199          * Generates unique ids. If the element already has an id, it is unchanged
200          * @param {String/HTMLElement/Element} el (optional) The element to generate an id for
201          * @param {String} prefix (optional) Id prefix (defaults "Roo-gen")
202          * @return {String} The generated Id.
203          */
204         id : function(el, prefix){
205             prefix = prefix || "roo-gen";
206             el = Roo.getDom(el);
207             var id = prefix + (++idSeed);
208             return el ? (el.id ? el.id : (el.id = id)) : id;
209         },
210          
211        
212         /**
213          * Extends one class with another class and optionally overrides members with the passed literal. This class
214          * also adds the function "override()" to the class that can be used to override
215          * members on an instance.
216          * @param {Object} subclass The class inheriting the functionality
217          * @param {Object} superclass The class being extended
218          * @param {Object} overrides (optional) A literal with members
219          * @method extend
220          */
221         extend : function(){
222             // inline overrides
223             var io = function(o){
224                 for(var m in o){
225                     this[m] = o[m];
226                 }
227             };
228             return function(sb, sp, overrides){
229                 if(typeof sp == 'object'){ // eg. prototype, rather than function constructor..
230                     overrides = sp;
231                     sp = sb;
232                     sb = function(){sp.apply(this, arguments);};
233                 }
234                 var F = function(){}, sbp, spp = sp.prototype;
235                 F.prototype = spp;
236                 sbp = sb.prototype = new F();
237                 sbp.constructor=sb;
238                 sb.superclass=spp;
239                 
240                 if(spp.constructor == Object.prototype.constructor){
241                     spp.constructor=sp;
242                    
243                 }
244                 
245                 sb.override = function(o){
246                     Roo.override(sb, o);
247                 };
248                 sbp.override = io;
249                 Roo.override(sb, overrides);
250                 return sb;
251             };
252         }(),
253
254         /**
255          * Adds a list of functions to the prototype of an existing class, overwriting any existing methods with the same name.
256          * Usage:<pre><code>
257 Roo.override(MyClass, {
258     newMethod1: function(){
259         // etc.
260     },
261     newMethod2: function(foo){
262         // etc.
263     }
264 });
265  </code></pre>
266          * @param {Object} origclass The class to override
267          * @param {Object} overrides The list of functions to add to origClass.  This should be specified as an object literal
268          * containing one or more methods.
269          * @method override
270          */
271         override : function(origclass, overrides){
272             if(overrides){
273                 var p = origclass.prototype;
274                 for(var method in overrides){
275                     p[method] = overrides[method];
276                 }
277             }
278         },
279         /**
280          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
281          * <pre><code>
282 Roo.namespace('Company', 'Company.data');
283 Company.Widget = function() { ... }
284 Company.data.CustomStore = function(config) { ... }
285 </code></pre>
286          * @param {String} namespace1
287          * @param {String} namespace2
288          * @param {String} etc
289          * @method namespace
290          */
291         namespace : function(){
292             var a=arguments, o=null, i, j, d, rt;
293             for (i=0; i<a.length; ++i) {
294                 d=a[i].split(".");
295                 rt = d[0];
296                 /** eval:var:o */
297                 eval('if (typeof ' + rt + ' == "undefined"){' + rt + ' = {};} o = ' + rt + ';');
298                 for (j=1; j<d.length; ++j) {
299                     o[d[j]]=o[d[j]] || {};
300                     o=o[d[j]];
301                 }
302             }
303         },
304         /**
305          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
306          * <pre><code>
307 Roo.factory({ xns: Roo.data, xtype : 'Store', .....});
308 Roo.factory(conf, Roo.data);
309 </code></pre>
310          * @param {String} classname
311          * @param {String} namespace (optional)
312          * @method factory
313          */
314          
315         factory : function(c, ns)
316         {
317             // no xtype, no ns or c.xns - or forced off by c.xns
318             if (!c.xtype   || (!ns && !c.xns) ||  (c.xns === false)) { // not enough info...
319                 return c;
320             }
321             ns = c.xns ? c.xns : ns; // if c.xns is set, then use that..
322             if (c.constructor == ns[c.xtype]) {// already created...
323                 return c;
324             }
325             if (ns[c.xtype]) {
326                 if (Roo.debug) { Roo.log("Roo.Factory(" + c.xtype + ")"); }
327                 var ret = new ns[c.xtype](c);
328                 ret.xns = false;
329                 return ret;
330             }
331             c.xns = false; // prevent recursion..
332             return c;
333         },
334          /**
335          * Logs to console if it can.
336          *
337          * @param {String|Object} string
338          * @method log
339          */
340         log : function(s)
341         {
342             if ((typeof(console) == 'undefined') || (typeof(console.log) == 'undefined')) {
343                 return; // alerT?
344             }
345             
346             console.log(s);
347         },
348         /**
349          * Takes an object and converts it to an encoded URL. e.g. Roo.urlEncode({foo: 1, bar: 2}); would return "foo=1&bar=2".  Optionally, property values can be arrays, instead of keys and the resulting string that's returned will contain a name/value pair for each array value.
350          * @param {Object} o
351          * @return {String}
352          */
353         urlEncode : function(o){
354             if(!o){
355                 return "";
356             }
357             var buf = [];
358             for(var key in o){
359                 var ov = o[key], k = Roo.encodeURIComponent(key);
360                 var type = typeof ov;
361                 if(type == 'undefined'){
362                     buf.push(k, "=&");
363                 }else if(type != "function" && type != "object"){
364                     buf.push(k, "=", Roo.encodeURIComponent(ov), "&");
365                 }else if(ov instanceof Array){
366                     if (ov.length) {
367                             for(var i = 0, len = ov.length; i < len; i++) {
368                                 buf.push(k, "=", Roo.encodeURIComponent(ov[i] === undefined ? '' : ov[i]), "&");
369                             }
370                         } else {
371                             buf.push(k, "=&");
372                         }
373                 }
374             }
375             buf.pop();
376             return buf.join("");
377         },
378          /**
379          * Safe version of encodeURIComponent
380          * @param {String} data 
381          * @return {String} 
382          */
383         
384         encodeURIComponent : function (data)
385         {
386             try {
387                 return encodeURIComponent(data);
388             } catch(e) {} // should be an uri encode error.
389             
390             if (data == '' || data == null){
391                return '';
392             }
393             // http://stackoverflow.com/questions/2596483/unicode-and-uri-encoding-decoding-and-escaping-in-javascript
394             function nibble_to_hex(nibble){
395                 var chars = '0123456789ABCDEF';
396                 return chars.charAt(nibble);
397             }
398             data = data.toString();
399             var buffer = '';
400             for(var i=0; i<data.length; i++){
401                 var c = data.charCodeAt(i);
402                 var bs = new Array();
403                 if (c > 0x10000){
404                         // 4 bytes
405                     bs[0] = 0xF0 | ((c & 0x1C0000) >>> 18);
406                     bs[1] = 0x80 | ((c & 0x3F000) >>> 12);
407                     bs[2] = 0x80 | ((c & 0xFC0) >>> 6);
408                     bs[3] = 0x80 | (c & 0x3F);
409                 }else if (c > 0x800){
410                          // 3 bytes
411                     bs[0] = 0xE0 | ((c & 0xF000) >>> 12);
412                     bs[1] = 0x80 | ((c & 0xFC0) >>> 6);
413                     bs[2] = 0x80 | (c & 0x3F);
414                 }else if (c > 0x80){
415                        // 2 bytes
416                     bs[0] = 0xC0 | ((c & 0x7C0) >>> 6);
417                     bs[1] = 0x80 | (c & 0x3F);
418                 }else{
419                         // 1 byte
420                     bs[0] = c;
421                 }
422                 for(var j=0; j<bs.length; j++){
423                     var b = bs[j];
424                     var hex = nibble_to_hex((b & 0xF0) >>> 4) 
425                             + nibble_to_hex(b &0x0F);
426                     buffer += '%'+hex;
427                }
428             }
429             return buffer;    
430              
431         },
432
433         /**
434          * Takes an encoded URL and and converts it to an object. e.g. Roo.urlDecode("foo=1&bar=2"); would return {foo: 1, bar: 2} or Roo.urlDecode("foo=1&bar=2&bar=3&bar=4", true); would return {foo: 1, bar: [2, 3, 4]}.
435          * @param {String} string
436          * @param {Boolean} overwrite (optional) Items of the same name will overwrite previous values instead of creating an an array (Defaults to false).
437          * @return {Object} A literal with members
438          */
439         urlDecode : function(string, overwrite){
440             if(!string || !string.length){
441                 return {};
442             }
443             var obj = {};
444             var pairs = string.split('&');
445             var pair, name, value;
446             for(var i = 0, len = pairs.length; i < len; i++){
447                 pair = pairs[i].split('=');
448                 name = decodeURIComponent(pair[0]);
449                 value = decodeURIComponent(pair[1]);
450                 if(overwrite !== true){
451                     if(typeof obj[name] == "undefined"){
452                         obj[name] = value;
453                     }else if(typeof obj[name] == "string"){
454                         obj[name] = [obj[name]];
455                         obj[name].push(value);
456                     }else{
457                         obj[name].push(value);
458                     }
459                 }else{
460                     obj[name] = value;
461                 }
462             }
463             return obj;
464         },
465
466         /**
467          * Iterates an array calling the passed function with each item, stopping if your function returns false. If the
468          * passed array is not really an array, your function is called once with it.
469          * The supplied function is called with (Object item, Number index, Array allItems).
470          * @param {Array/NodeList/Mixed} array
471          * @param {Function} fn
472          * @param {Object} scope
473          */
474         each : function(array, fn, scope){
475             if(typeof array.length == "undefined" || typeof array == "string"){
476                 array = [array];
477             }
478             for(var i = 0, len = array.length; i < len; i++){
479                 if(fn.call(scope || array[i], array[i], i, array) === false){ return i; };
480             }
481         },
482
483         // deprecated
484         combine : function(){
485             var as = arguments, l = as.length, r = [];
486             for(var i = 0; i < l; i++){
487                 var a = as[i];
488                 if(a instanceof Array){
489                     r = r.concat(a);
490                 }else if(a.length !== undefined && !a.substr){
491                     r = r.concat(Array.prototype.slice.call(a, 0));
492                 }else{
493                     r.push(a);
494                 }
495             }
496             return r;
497         },
498
499         /**
500          * Escapes the passed string for use in a regular expression
501          * @param {String} str
502          * @return {String}
503          */
504         escapeRe : function(s) {
505             return s.replace(/([.*+?^${}()|[\]\/\\])/g, "\\$1");
506         },
507
508         // internal
509         callback : function(cb, scope, args, delay){
510             if(typeof cb == "function"){
511                 if(delay){
512                     cb.defer(delay, scope, args || []);
513                 }else{
514                     cb.apply(scope, args || []);
515                 }
516             }
517         },
518
519         /**
520          * Return the dom node for the passed string (id), dom node, or Roo.Element
521          * @param {String/HTMLElement/Roo.Element} el
522          * @return HTMLElement
523          */
524         getDom : function(el){
525             if(!el){
526                 return null;
527             }
528             return el.dom ? el.dom : (typeof el == 'string' ? document.getElementById(el) : el);
529         },
530
531         /**
532         * Shorthand for {@link Roo.ComponentMgr#get}
533         * @param {String} id
534         * @return Roo.Component
535         */
536         getCmp : function(id){
537             return Roo.ComponentMgr.get(id);
538         },
539          
540         num : function(v, defaultValue){
541             if(typeof v != 'number'){
542                 return defaultValue;
543             }
544             return v;
545         },
546
547         destroy : function(){
548             for(var i = 0, a = arguments, len = a.length; i < len; i++) {
549                 var as = a[i];
550                 if(as){
551                     if(as.dom){
552                         as.removeAllListeners();
553                         as.remove();
554                         continue;
555                     }
556                     if(typeof as.purgeListeners == 'function'){
557                         as.purgeListeners();
558                     }
559                     if(typeof as.destroy == 'function'){
560                         as.destroy();
561                     }
562                 }
563             }
564         },
565
566         // inpired by a similar function in mootools library
567         /**
568          * Returns the type of object that is passed in. If the object passed in is null or undefined it
569          * return false otherwise it returns one of the following values:<ul>
570          * <li><b>string</b>: If the object passed is a string</li>
571          * <li><b>number</b>: If the object passed is a number</li>
572          * <li><b>boolean</b>: If the object passed is a boolean value</li>
573          * <li><b>function</b>: If the object passed is a function reference</li>
574          * <li><b>object</b>: If the object passed is an object</li>
575          * <li><b>array</b>: If the object passed is an array</li>
576          * <li><b>regexp</b>: If the object passed is a regular expression</li>
577          * <li><b>element</b>: If the object passed is a DOM Element</li>
578          * <li><b>nodelist</b>: If the object passed is a DOM NodeList</li>
579          * <li><b>textnode</b>: If the object passed is a DOM text node and contains something other than whitespace</li>
580          * <li><b>whitespace</b>: If the object passed is a DOM text node and contains only whitespace</li>
581          * @param {Mixed} object
582          * @return {String}
583          */
584         type : function(o){
585             if(o === undefined || o === null){
586                 return false;
587             }
588             if(o.htmlElement){
589                 return 'element';
590             }
591             var t = typeof o;
592             if(t == 'object' && o.nodeName) {
593                 switch(o.nodeType) {
594                     case 1: return 'element';
595                     case 3: return (/\S/).test(o.nodeValue) ? 'textnode' : 'whitespace';
596                 }
597             }
598             if(t == 'object' || t == 'function') {
599                 switch(o.constructor) {
600                     case Array: return 'array';
601                     case RegExp: return 'regexp';
602                 }
603                 if(typeof o.length == 'number' && typeof o.item == 'function') {
604                     return 'nodelist';
605                 }
606             }
607             return t;
608         },
609
610         /**
611          * Returns true if the passed value is null, undefined or an empty string (optional).
612          * @param {Mixed} value The value to test
613          * @param {Boolean} allowBlank (optional) Pass true if an empty string is not considered empty
614          * @return {Boolean}
615          */
616         isEmpty : function(v, allowBlank){
617             return v === null || v === undefined || (!allowBlank ? v === '' : false);
618         },
619         
620         /** @type Boolean */
621         isOpera : isOpera,
622         /** @type Boolean */
623         isSafari : isSafari,
624         /** @type Boolean */
625         isFirefox : isFirefox,
626         /** @type Boolean */
627         isIE : isIE,
628         /** @type Boolean */
629         isIE7 : isIE7,
630         /** @type Boolean */
631         isIE11 : isIE11,
632         /** @type Boolean */
633         isEdge : isEdge,
634         /** @type Boolean */
635         isGecko : isGecko,
636         /** @type Boolean */
637         isBorderBox : isBorderBox,
638         /** @type Boolean */
639         isWindows : isWindows,
640         /** @type Boolean */
641         isLinux : isLinux,
642         /** @type Boolean */
643         isMac : isMac,
644         /** @type Boolean */
645         isIOS : isIOS,
646         /** @type Boolean */
647         isAndroid : isAndroid,
648         /** @type Boolean */
649         isTouch : isTouch,
650
651         /**
652          * By default, Ext intelligently decides whether floating elements should be shimmed. If you are using flash,
653          * you may want to set this to true.
654          * @type Boolean
655          */
656         useShims : ((isIE && !isIE7) || (isGecko && isMac)),
657         
658         
659                 
660         /**
661          * Selects a single element as a Roo Element
662          * This is about as close as you can get to jQuery's $('do crazy stuff')
663          * @param {String} selector The selector/xpath query
664          * @param {Node} root (optional) The start of the query (defaults to document).
665          * @return {Roo.Element}
666          */
667         selectNode : function(selector, root) 
668         {
669             var node = Roo.DomQuery.selectNode(selector,root);
670             return node ? Roo.get(node) : new Roo.Element(false);
671         }
672         
673     });
674
675
676 })();
677
678 Roo.namespace("Roo", "Roo.util", "Roo.grid", "Roo.dd", "Roo.tree", "Roo.data",
679                 "Roo.form", "Roo.menu", "Roo.state", "Roo.lib", "Roo.layout",
680                 "Roo.app", "Roo.ux",
681                 "Roo.bootstrap",
682                 "Roo.bootstrap.dash");
683 /*
684  * Based on:
685  * Ext JS Library 1.1.1
686  * Copyright(c) 2006-2007, Ext JS, LLC.
687  *
688  * Originally Released Under LGPL - original licence link has changed is not relivant.
689  *
690  * Fork - LGPL
691  * <script type="text/javascript">
692  */
693
694 (function() {    
695     // wrappedn so fnCleanup is not in global scope...
696     if(Roo.isIE) {
697         function fnCleanUp() {
698             var p = Function.prototype;
699             delete p.createSequence;
700             delete p.defer;
701             delete p.createDelegate;
702             delete p.createCallback;
703             delete p.createInterceptor;
704
705             window.detachEvent("onunload", fnCleanUp);
706         }
707         window.attachEvent("onunload", fnCleanUp);
708     }
709 })();
710
711
712 /**
713  * @class Function
714  * These functions are available on every Function object (any JavaScript function).
715  */
716 Roo.apply(Function.prototype, {
717      /**
718      * Creates a callback that passes arguments[0], arguments[1], arguments[2], ...
719      * Call directly on any function. Example: <code>myFunction.createCallback(myarg, myarg2)</code>
720      * Will create a function that is bound to those 2 args.
721      * @return {Function} The new function
722     */
723     createCallback : function(/*args...*/){
724         // make args available, in function below
725         var args = arguments;
726         var method = this;
727         return function() {
728             return method.apply(window, args);
729         };
730     },
731
732     /**
733      * Creates a delegate (callback) that sets the scope to obj.
734      * Call directly on any function. Example: <code>this.myFunction.createDelegate(this)</code>
735      * Will create a function that is automatically scoped to this.
736      * @param {Object} obj (optional) The object for which the scope is set
737      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
738      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
739      *                                             if a number the args are inserted at the specified position
740      * @return {Function} The new function
741      */
742     createDelegate : function(obj, args, appendArgs){
743         var method = this;
744         return function() {
745             var callArgs = args || arguments;
746             if(appendArgs === true){
747                 callArgs = Array.prototype.slice.call(arguments, 0);
748                 callArgs = callArgs.concat(args);
749             }else if(typeof appendArgs == "number"){
750                 callArgs = Array.prototype.slice.call(arguments, 0); // copy arguments first
751                 var applyArgs = [appendArgs, 0].concat(args); // create method call params
752                 Array.prototype.splice.apply(callArgs, applyArgs); // splice them in
753             }
754             return method.apply(obj || window, callArgs);
755         };
756     },
757
758     /**
759      * Calls this function after the number of millseconds specified.
760      * @param {Number} millis The number of milliseconds for the setTimeout call (if 0 the function is executed immediately)
761      * @param {Object} obj (optional) The object for which the scope is set
762      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
763      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
764      *                                             if a number the args are inserted at the specified position
765      * @return {Number} The timeout id that can be used with clearTimeout
766      */
767     defer : function(millis, obj, args, appendArgs){
768         var fn = this.createDelegate(obj, args, appendArgs);
769         if(millis){
770             return setTimeout(fn, millis);
771         }
772         fn();
773         return 0;
774     },
775     /**
776      * Create a combined function call sequence of the original function + the passed function.
777      * The resulting function returns the results of the original function.
778      * The passed fcn is called with the parameters of the original function
779      * @param {Function} fcn The function to sequence
780      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
781      * @return {Function} The new function
782      */
783     createSequence : function(fcn, scope){
784         if(typeof fcn != "function"){
785             return this;
786         }
787         var method = this;
788         return function() {
789             var retval = method.apply(this || window, arguments);
790             fcn.apply(scope || this || window, arguments);
791             return retval;
792         };
793     },
794
795     /**
796      * Creates an interceptor function. The passed fcn is called before the original one. If it returns false, the original one is not called.
797      * The resulting function returns the results of the original function.
798      * The passed fcn is called with the parameters of the original function.
799      * @addon
800      * @param {Function} fcn The function to call before the original
801      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
802      * @return {Function} The new function
803      */
804     createInterceptor : function(fcn, scope){
805         if(typeof fcn != "function"){
806             return this;
807         }
808         var method = this;
809         return function() {
810             fcn.target = this;
811             fcn.method = method;
812             if(fcn.apply(scope || this || window, arguments) === false){
813                 return;
814             }
815             return method.apply(this || window, arguments);
816         };
817     }
818 });
819 /*
820  * Based on:
821  * Ext JS Library 1.1.1
822  * Copyright(c) 2006-2007, Ext JS, LLC.
823  *
824  * Originally Released Under LGPL - original licence link has changed is not relivant.
825  *
826  * Fork - LGPL
827  * <script type="text/javascript">
828  */
829
830 Roo.applyIf(String, {
831     
832     /** @scope String */
833     
834     /**
835      * Escapes the passed string for ' and \
836      * @param {String} string The string to escape
837      * @return {String} The escaped string
838      * @static
839      */
840     escape : function(string) {
841         return string.replace(/('|\\)/g, "\\$1");
842     },
843
844     /**
845      * Pads the left side of a string with a specified character.  This is especially useful
846      * for normalizing number and date strings.  Example usage:
847      * <pre><code>
848 var s = String.leftPad('123', 5, '0');
849 // s now contains the string: '00123'
850 </code></pre>
851      * @param {String} string The original string
852      * @param {Number} size The total length of the output string
853      * @param {String} char (optional) The character with which to pad the original string (defaults to empty string " ")
854      * @return {String} The padded string
855      * @static
856      */
857     leftPad : function (val, size, ch) {
858         var result = new String(val);
859         if(ch === null || ch === undefined || ch === '') {
860             ch = " ";
861         }
862         while (result.length < size) {
863             result = ch + result;
864         }
865         return result;
866     },
867
868     /**
869      * Allows you to define a tokenized string and pass an arbitrary number of arguments to replace the tokens.  Each
870      * token must be unique, and must increment in the format {0}, {1}, etc.  Example usage:
871      * <pre><code>
872 var cls = 'my-class', text = 'Some text';
873 var s = String.format('<div class="{0}">{1}</div>', cls, text);
874 // s now contains the string: '<div class="my-class">Some text</div>'
875 </code></pre>
876      * @param {String} string The tokenized string to be formatted
877      * @param {String} value1 The value to replace token {0}
878      * @param {String} value2 Etc...
879      * @return {String} The formatted string
880      * @static
881      */
882     format : function(format){
883         var args = Array.prototype.slice.call(arguments, 1);
884         return format.replace(/\{(\d+)\}/g, function(m, i){
885             return Roo.util.Format.htmlEncode(args[i]);
886         });
887     }
888   
889     
890 });
891
892 /**
893  * Utility function that allows you to easily switch a string between two alternating values.  The passed value
894  * is compared to the current string, and if they are equal, the other value that was passed in is returned.  If
895  * they are already different, the first value passed in is returned.  Note that this method returns the new value
896  * but does not change the current string.
897  * <pre><code>
898 // alternate sort directions
899 sort = sort.toggle('ASC', 'DESC');
900
901 // instead of conditional logic:
902 sort = (sort == 'ASC' ? 'DESC' : 'ASC');
903 </code></pre>
904  * @param {String} value The value to compare to the current string
905  * @param {String} other The new value to use if the string already equals the first value passed in
906  * @return {String} The new value
907  */
908  
909 String.prototype.toggle = function(value, other){
910     return this == value ? other : value;
911 };
912
913
914 /**
915   * Remove invalid unicode characters from a string 
916   *
917   * @return {String} The clean string
918   */
919 String.prototype.unicodeClean = function () {
920     return this.replace(/[\s\S]/g,
921         function(character) {
922             if (character.charCodeAt()< 256) {
923               return character;
924            }
925            try {
926                 encodeURIComponent(character);
927            } catch(e) { 
928               return '';
929            }
930            return character;
931         }
932     );
933 };
934   
935 /*
936  * Based on:
937  * Ext JS Library 1.1.1
938  * Copyright(c) 2006-2007, Ext JS, LLC.
939  *
940  * Originally Released Under LGPL - original licence link has changed is not relivant.
941  *
942  * Fork - LGPL
943  * <script type="text/javascript">
944  */
945
946  /**
947  * @class Number
948  */
949 Roo.applyIf(Number.prototype, {
950     /**
951      * Checks whether or not the current number is within a desired range.  If the number is already within the
952      * range it is returned, otherwise the min or max value is returned depending on which side of the range is
953      * exceeded.  Note that this method returns the constrained value but does not change the current number.
954      * @param {Number} min The minimum number in the range
955      * @param {Number} max The maximum number in the range
956      * @return {Number} The constrained value if outside the range, otherwise the current value
957      */
958     constrain : function(min, max){
959         return Math.min(Math.max(this, min), max);
960     }
961 });/*
962  * Based on:
963  * Ext JS Library 1.1.1
964  * Copyright(c) 2006-2007, Ext JS, LLC.
965  *
966  * Originally Released Under LGPL - original licence link has changed is not relivant.
967  *
968  * Fork - LGPL
969  * <script type="text/javascript">
970  */
971  /**
972  * @class Array
973  */
974 Roo.applyIf(Array.prototype, {
975     /**
976      * 
977      * Checks whether or not the specified object exists in the array.
978      * @param {Object} o The object to check for
979      * @return {Number} The index of o in the array (or -1 if it is not found)
980      */
981     indexOf : function(o){
982        for (var i = 0, len = this.length; i < len; i++){
983               if(this[i] == o) { return i; }
984        }
985            return -1;
986     },
987
988     /**
989      * Removes the specified object from the array.  If the object is not found nothing happens.
990      * @param {Object} o The object to remove
991      */
992     remove : function(o){
993        var index = this.indexOf(o);
994        if(index != -1){
995            this.splice(index, 1);
996        }
997     },
998     /**
999      * Map (JS 1.6 compatibility)
1000      * @param {Function} function  to call
1001      */
1002     map : function(fun )
1003     {
1004         var len = this.length >>> 0;
1005         if (typeof fun != "function") {
1006             throw new TypeError();
1007         }
1008         var res = new Array(len);
1009         var thisp = arguments[1];
1010         for (var i = 0; i < len; i++)
1011         {
1012             if (i in this) {
1013                 res[i] = fun.call(thisp, this[i], i, this);
1014             }
1015         }
1016
1017         return res;
1018     }
1019     
1020 });
1021
1022
1023  
1024 /*
1025  * Based on:
1026  * Ext JS Library 1.1.1
1027  * Copyright(c) 2006-2007, Ext JS, LLC.
1028  *
1029  * Originally Released Under LGPL - original licence link has changed is not relivant.
1030  *
1031  * Fork - LGPL
1032  * <script type="text/javascript">
1033  */
1034
1035 /**
1036  * @class Date
1037  *
1038  * The date parsing and format syntax is a subset of
1039  * <a href="http://www.php.net/date">PHP's date() function</a>, and the formats that are
1040  * supported will provide results equivalent to their PHP versions.
1041  *
1042  * Following is the list of all currently supported formats:
1043  *<pre>
1044 Sample date:
1045 'Wed Jan 10 2007 15:05:01 GMT-0600 (Central Standard Time)'
1046
1047 Format  Output      Description
1048 ------  ----------  --------------------------------------------------------------
1049   d      10         Day of the month, 2 digits with leading zeros
1050   D      Wed        A textual representation of a day, three letters
1051   j      10         Day of the month without leading zeros
1052   l      Wednesday  A full textual representation of the day of the week
1053   S      th         English ordinal day of month suffix, 2 chars (use with j)
1054   w      3          Numeric representation of the day of the week
1055   z      9          The julian date, or day of the year (0-365)
1056   W      01         ISO-8601 2-digit week number of year, weeks starting on Monday (00-52)
1057   F      January    A full textual representation of the month
1058   m      01         Numeric representation of a month, with leading zeros
1059   M      Jan        Month name abbreviation, three letters
1060   n      1          Numeric representation of a month, without leading zeros
1061   t      31         Number of days in the given month
1062   L      0          Whether it's a leap year (1 if it is a leap year, else 0)
1063   Y      2007       A full numeric representation of a year, 4 digits
1064   y      07         A two digit representation of a year
1065   a      pm         Lowercase Ante meridiem and Post meridiem
1066   A      PM         Uppercase Ante meridiem and Post meridiem
1067   g      3          12-hour format of an hour without leading zeros
1068   G      15         24-hour format of an hour without leading zeros
1069   h      03         12-hour format of an hour with leading zeros
1070   H      15         24-hour format of an hour with leading zeros
1071   i      05         Minutes with leading zeros
1072   s      01         Seconds, with leading zeros
1073   O      -0600      Difference to Greenwich time (GMT) in hours (Allows +08, without minutes)
1074   P      -06:00     Difference to Greenwich time (GMT) with colon between hours and minutes
1075   T      CST        Timezone setting of the machine running the code
1076   Z      -21600     Timezone offset in seconds (negative if west of UTC, positive if east)
1077 </pre>
1078  *
1079  * Example usage (note that you must escape format specifiers with '\\' to render them as character literals):
1080  * <pre><code>
1081 var dt = new Date('1/10/2007 03:05:01 PM GMT-0600');
1082 document.write(dt.format('Y-m-d'));                         //2007-01-10
1083 document.write(dt.format('F j, Y, g:i a'));                 //January 10, 2007, 3:05 pm
1084 document.write(dt.format('l, \\t\\he dS of F Y h:i:s A'));  //Wednesday, the 10th of January 2007 03:05:01 PM
1085  </code></pre>
1086  *
1087  * Here are some standard date/time patterns that you might find helpful.  They
1088  * are not part of the source of Date.js, but to use them you can simply copy this
1089  * block of code into any script that is included after Date.js and they will also become
1090  * globally available on the Date object.  Feel free to add or remove patterns as needed in your code.
1091  * <pre><code>
1092 Date.patterns = {
1093     ISO8601Long:"Y-m-d H:i:s",
1094     ISO8601Short:"Y-m-d",
1095     ShortDate: "n/j/Y",
1096     LongDate: "l, F d, Y",
1097     FullDateTime: "l, F d, Y g:i:s A",
1098     MonthDay: "F d",
1099     ShortTime: "g:i A",
1100     LongTime: "g:i:s A",
1101     SortableDateTime: "Y-m-d\\TH:i:s",
1102     UniversalSortableDateTime: "Y-m-d H:i:sO",
1103     YearMonth: "F, Y"
1104 };
1105 </code></pre>
1106  *
1107  * Example usage:
1108  * <pre><code>
1109 var dt = new Date();
1110 document.write(dt.format(Date.patterns.ShortDate));
1111  </code></pre>
1112  */
1113
1114 /*
1115  * Most of the date-formatting functions below are the excellent work of Baron Schwartz.
1116  * They generate precompiled functions from date formats instead of parsing and
1117  * processing the pattern every time you format a date.  These functions are available
1118  * on every Date object (any javascript function).
1119  *
1120  * The original article and download are here:
1121  * http://www.xaprb.com/blog/2005/12/12/javascript-closures-for-runtime-efficiency/
1122  *
1123  */
1124  
1125  
1126  // was in core
1127 /**
1128  Returns the number of milliseconds between this date and date
1129  @param {Date} date (optional) Defaults to now
1130  @return {Number} The diff in milliseconds
1131  @member Date getElapsed
1132  */
1133 Date.prototype.getElapsed = function(date) {
1134         return Math.abs((date || new Date()).getTime()-this.getTime());
1135 };
1136 // was in date file..
1137
1138
1139 // private
1140 Date.parseFunctions = {count:0};
1141 // private
1142 Date.parseRegexes = [];
1143 // private
1144 Date.formatFunctions = {count:0};
1145
1146 // private
1147 Date.prototype.dateFormat = function(format) {
1148     if (Date.formatFunctions[format] == null) {
1149         Date.createNewFormat(format);
1150     }
1151     var func = Date.formatFunctions[format];
1152     return this[func]();
1153 };
1154
1155
1156 /**
1157  * Formats a date given the supplied format string
1158  * @param {String} format The format string
1159  * @return {String} The formatted date
1160  * @method
1161  */
1162 Date.prototype.format = Date.prototype.dateFormat;
1163
1164 // private
1165 Date.createNewFormat = function(format) {
1166     var funcName = "format" + Date.formatFunctions.count++;
1167     Date.formatFunctions[format] = funcName;
1168     var code = "Date.prototype." + funcName + " = function(){return ";
1169     var special = false;
1170     var ch = '';
1171     for (var i = 0; i < format.length; ++i) {
1172         ch = format.charAt(i);
1173         if (!special && ch == "\\") {
1174             special = true;
1175         }
1176         else if (special) {
1177             special = false;
1178             code += "'" + String.escape(ch) + "' + ";
1179         }
1180         else {
1181             code += Date.getFormatCode(ch);
1182         }
1183     }
1184     /** eval:var:zzzzzzzzzzzzz */
1185     eval(code.substring(0, code.length - 3) + ";}");
1186 };
1187
1188 // private
1189 Date.getFormatCode = function(character) {
1190     switch (character) {
1191     case "d":
1192         return "String.leftPad(this.getDate(), 2, '0') + ";
1193     case "D":
1194         return "Date.dayNames[this.getDay()].substring(0, 3) + ";
1195     case "j":
1196         return "this.getDate() + ";
1197     case "l":
1198         return "Date.dayNames[this.getDay()] + ";
1199     case "S":
1200         return "this.getSuffix() + ";
1201     case "w":
1202         return "this.getDay() + ";
1203     case "z":
1204         return "this.getDayOfYear() + ";
1205     case "W":
1206         return "this.getWeekOfYear() + ";
1207     case "F":
1208         return "Date.monthNames[this.getMonth()] + ";
1209     case "m":
1210         return "String.leftPad(this.getMonth() + 1, 2, '0') + ";
1211     case "M":
1212         return "Date.monthNames[this.getMonth()].substring(0, 3) + ";
1213     case "n":
1214         return "(this.getMonth() + 1) + ";
1215     case "t":
1216         return "this.getDaysInMonth() + ";
1217     case "L":
1218         return "(this.isLeapYear() ? 1 : 0) + ";
1219     case "Y":
1220         return "this.getFullYear() + ";
1221     case "y":
1222         return "('' + this.getFullYear()).substring(2, 4) + ";
1223     case "a":
1224         return "(this.getHours() < 12 ? 'am' : 'pm') + ";
1225     case "A":
1226         return "(this.getHours() < 12 ? 'AM' : 'PM') + ";
1227     case "g":
1228         return "((this.getHours() % 12) ? this.getHours() % 12 : 12) + ";
1229     case "G":
1230         return "this.getHours() + ";
1231     case "h":
1232         return "String.leftPad((this.getHours() % 12) ? this.getHours() % 12 : 12, 2, '0') + ";
1233     case "H":
1234         return "String.leftPad(this.getHours(), 2, '0') + ";
1235     case "i":
1236         return "String.leftPad(this.getMinutes(), 2, '0') + ";
1237     case "s":
1238         return "String.leftPad(this.getSeconds(), 2, '0') + ";
1239     case "O":
1240         return "this.getGMTOffset() + ";
1241     case "P":
1242         return "this.getGMTColonOffset() + ";
1243     case "T":
1244         return "this.getTimezone() + ";
1245     case "Z":
1246         return "(this.getTimezoneOffset() * -60) + ";
1247     default:
1248         return "'" + String.escape(character) + "' + ";
1249     }
1250 };
1251
1252 /**
1253  * Parses the passed string using the specified format. Note that this function expects dates in normal calendar
1254  * format, meaning that months are 1-based (1 = January) and not zero-based like in JavaScript dates.  Any part of
1255  * the date format that is not specified will default to the current date value for that part.  Time parts can also
1256  * be specified, but default to 0.  Keep in mind that the input date string must precisely match the specified format
1257  * string or the parse operation will fail.
1258  * Example Usage:
1259 <pre><code>
1260 //dt = Fri May 25 2007 (current date)
1261 var dt = new Date();
1262
1263 //dt = Thu May 25 2006 (today's month/day in 2006)
1264 dt = Date.parseDate("2006", "Y");
1265
1266 //dt = Sun Jan 15 2006 (all date parts specified)
1267 dt = Date.parseDate("2006-1-15", "Y-m-d");
1268
1269 //dt = Sun Jan 15 2006 15:20:01 GMT-0600 (CST)
1270 dt = Date.parseDate("2006-1-15 3:20:01 PM", "Y-m-d h:i:s A" );
1271 </code></pre>
1272  * @param {String} input The unparsed date as a string
1273  * @param {String} format The format the date is in
1274  * @return {Date} The parsed date
1275  * @static
1276  */
1277 Date.parseDate = function(input, format) {
1278     if (Date.parseFunctions[format] == null) {
1279         Date.createParser(format);
1280     }
1281     var func = Date.parseFunctions[format];
1282     return Date[func](input);
1283 };
1284 /**
1285  * @private
1286  */
1287
1288 Date.createParser = function(format) {
1289     var funcName = "parse" + Date.parseFunctions.count++;
1290     var regexNum = Date.parseRegexes.length;
1291     var currentGroup = 1;
1292     Date.parseFunctions[format] = funcName;
1293
1294     var code = "Date." + funcName + " = function(input){\n"
1295         + "var y = -1, m = -1, d = -1, h = -1, i = -1, s = -1, o, z, v;\n"
1296         + "var d = new Date();\n"
1297         + "y = d.getFullYear();\n"
1298         + "m = d.getMonth();\n"
1299         + "d = d.getDate();\n"
1300         + "if (typeof(input) !== 'string') { input = input.toString(); }\n"
1301         + "var results = input.match(Date.parseRegexes[" + regexNum + "]);\n"
1302         + "if (results && results.length > 0) {";
1303     var regex = "";
1304
1305     var special = false;
1306     var ch = '';
1307     for (var i = 0; i < format.length; ++i) {
1308         ch = format.charAt(i);
1309         if (!special && ch == "\\") {
1310             special = true;
1311         }
1312         else if (special) {
1313             special = false;
1314             regex += String.escape(ch);
1315         }
1316         else {
1317             var obj = Date.formatCodeToRegex(ch, currentGroup);
1318             currentGroup += obj.g;
1319             regex += obj.s;
1320             if (obj.g && obj.c) {
1321                 code += obj.c;
1322             }
1323         }
1324     }
1325
1326     code += "if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0 && s >= 0)\n"
1327         + "{v = new Date(y, m, d, h, i, s);}\n"
1328         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0)\n"
1329         + "{v = new Date(y, m, d, h, i);}\n"
1330         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0)\n"
1331         + "{v = new Date(y, m, d, h);}\n"
1332         + "else if (y >= 0 && m >= 0 && d > 0)\n"
1333         + "{v = new Date(y, m, d);}\n"
1334         + "else if (y >= 0 && m >= 0)\n"
1335         + "{v = new Date(y, m);}\n"
1336         + "else if (y >= 0)\n"
1337         + "{v = new Date(y);}\n"
1338         + "}return (v && (z || o))?\n" // favour UTC offset over GMT offset
1339         + "    ((z)? v.add(Date.SECOND, (v.getTimezoneOffset() * 60) + (z*1)) :\n" // reset to UTC, then add offset
1340         + "        v.add(Date.HOUR, (v.getGMTOffset() / 100) + (o / -100))) : v\n" // reset to GMT, then add offset
1341         + ";}";
1342
1343     Date.parseRegexes[regexNum] = new RegExp("^" + regex + "$");
1344     /** eval:var:zzzzzzzzzzzzz */
1345     eval(code);
1346 };
1347
1348 // private
1349 Date.formatCodeToRegex = function(character, currentGroup) {
1350     switch (character) {
1351     case "D":
1352         return {g:0,
1353         c:null,
1354         s:"(?:Sun|Mon|Tue|Wed|Thu|Fri|Sat)"};
1355     case "j":
1356         return {g:1,
1357             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1358             s:"(\\d{1,2})"}; // day of month without leading zeroes
1359     case "d":
1360         return {g:1,
1361             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1362             s:"(\\d{2})"}; // day of month with leading zeroes
1363     case "l":
1364         return {g:0,
1365             c:null,
1366             s:"(?:" + Date.dayNames.join("|") + ")"};
1367     case "S":
1368         return {g:0,
1369             c:null,
1370             s:"(?:st|nd|rd|th)"};
1371     case "w":
1372         return {g:0,
1373             c:null,
1374             s:"\\d"};
1375     case "z":
1376         return {g:0,
1377             c:null,
1378             s:"(?:\\d{1,3})"};
1379     case "W":
1380         return {g:0,
1381             c:null,
1382             s:"(?:\\d{2})"};
1383     case "F":
1384         return {g:1,
1385             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "].substring(0, 3)], 10);\n",
1386             s:"(" + Date.monthNames.join("|") + ")"};
1387     case "M":
1388         return {g:1,
1389             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "]], 10);\n",
1390             s:"(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)"};
1391     case "n":
1392         return {g:1,
1393             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1394             s:"(\\d{1,2})"}; // Numeric representation of a month, without leading zeros
1395     case "m":
1396         return {g:1,
1397             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1398             s:"(\\d{2})"}; // Numeric representation of a month, with leading zeros
1399     case "t":
1400         return {g:0,
1401             c:null,
1402             s:"\\d{1,2}"};
1403     case "L":
1404         return {g:0,
1405             c:null,
1406             s:"(?:1|0)"};
1407     case "Y":
1408         return {g:1,
1409             c:"y = parseInt(results[" + currentGroup + "], 10);\n",
1410             s:"(\\d{4})"};
1411     case "y":
1412         return {g:1,
1413             c:"var ty = parseInt(results[" + currentGroup + "], 10);\n"
1414                 + "y = ty > Date.y2kYear ? 1900 + ty : 2000 + ty;\n",
1415             s:"(\\d{1,2})"};
1416     case "a":
1417         return {g:1,
1418             c:"if (results[" + currentGroup + "] == 'am') {\n"
1419                 + "if (h == 12) { h = 0; }\n"
1420                 + "} else { if (h < 12) { h += 12; }}",
1421             s:"(am|pm)"};
1422     case "A":
1423         return {g:1,
1424             c:"if (results[" + currentGroup + "] == 'AM') {\n"
1425                 + "if (h == 12) { h = 0; }\n"
1426                 + "} else { if (h < 12) { h += 12; }}",
1427             s:"(AM|PM)"};
1428     case "g":
1429     case "G":
1430         return {g:1,
1431             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1432             s:"(\\d{1,2})"}; // 12/24-hr format  format of an hour without leading zeroes
1433     case "h":
1434     case "H":
1435         return {g:1,
1436             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1437             s:"(\\d{2})"}; //  12/24-hr format  format of an hour with leading zeroes
1438     case "i":
1439         return {g:1,
1440             c:"i = parseInt(results[" + currentGroup + "], 10);\n",
1441             s:"(\\d{2})"};
1442     case "s":
1443         return {g:1,
1444             c:"s = parseInt(results[" + currentGroup + "], 10);\n",
1445             s:"(\\d{2})"};
1446     case "O":
1447         return {g:1,
1448             c:[
1449                 "o = results[", currentGroup, "];\n",
1450                 "var sn = o.substring(0,1);\n", // get + / - sign
1451                 "var hr = o.substring(1,3)*1 + Math.floor(o.substring(3,5) / 60);\n", // get hours (performs minutes-to-hour conversion also)
1452                 "var mn = o.substring(3,5) % 60;\n", // get minutes
1453                 "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n", // -12hrs <= GMT offset <= 14hrs
1454                 "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1455             ].join(""),
1456             s:"([+\-]\\d{2,4})"};
1457     
1458     
1459     case "P":
1460         return {g:1,
1461                 c:[
1462                    "o = results[", currentGroup, "];\n",
1463                    "var sn = o.substring(0,1);\n",
1464                    "var hr = o.substring(1,3)*1 + Math.floor(o.substring(4,6) / 60);\n",
1465                    "var mn = o.substring(4,6) % 60;\n",
1466                    "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n",
1467                         "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1468             ].join(""),
1469             s:"([+\-]\\d{4})"};
1470     case "T":
1471         return {g:0,
1472             c:null,
1473             s:"[A-Z]{1,4}"}; // timezone abbrev. may be between 1 - 4 chars
1474     case "Z":
1475         return {g:1,
1476             c:"z = results[" + currentGroup + "];\n" // -43200 <= UTC offset <= 50400
1477                   + "z = (-43200 <= z*1 && z*1 <= 50400)? z : null;\n",
1478             s:"([+\-]?\\d{1,5})"}; // leading '+' sign is optional for UTC offset
1479     default:
1480         return {g:0,
1481             c:null,
1482             s:String.escape(character)};
1483     }
1484 };
1485
1486 /**
1487  * Get the timezone abbreviation of the current date (equivalent to the format specifier 'T').
1488  * @return {String} The abbreviated timezone name (e.g. 'CST')
1489  */
1490 Date.prototype.getTimezone = function() {
1491     return this.toString().replace(/^.*? ([A-Z]{1,4})[\-+][0-9]{4} .*$/, "$1");
1492 };
1493
1494 /**
1495  * Get the offset from GMT of the current date (equivalent to the format specifier 'O').
1496  * @return {String} The 4-character offset string prefixed with + or - (e.g. '-0600')
1497  */
1498 Date.prototype.getGMTOffset = function() {
1499     return (this.getTimezoneOffset() > 0 ? "-" : "+")
1500         + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1501         + String.leftPad(this.getTimezoneOffset() % 60, 2, "0");
1502 };
1503
1504 /**
1505  * Get the offset from GMT of the current date (equivalent to the format specifier 'P').
1506  * @return {String} 2-characters representing hours and 2-characters representing minutes
1507  * seperated by a colon and prefixed with + or - (e.g. '-06:00')
1508  */
1509 Date.prototype.getGMTColonOffset = function() {
1510         return (this.getTimezoneOffset() > 0 ? "-" : "+")
1511                 + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1512                 + ":"
1513                 + String.leftPad(this.getTimezoneOffset() %60, 2, "0");
1514 }
1515
1516 /**
1517  * Get the numeric day number of the year, adjusted for leap year.
1518  * @return {Number} 0 through 364 (365 in leap years)
1519  */
1520 Date.prototype.getDayOfYear = function() {
1521     var num = 0;
1522     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1523     for (var i = 0; i < this.getMonth(); ++i) {
1524         num += Date.daysInMonth[i];
1525     }
1526     return num + this.getDate() - 1;
1527 };
1528
1529 /**
1530  * Get the string representation of the numeric week number of the year
1531  * (equivalent to the format specifier 'W').
1532  * @return {String} '00' through '52'
1533  */
1534 Date.prototype.getWeekOfYear = function() {
1535     // Skip to Thursday of this week
1536     var now = this.getDayOfYear() + (4 - this.getDay());
1537     // Find the first Thursday of the year
1538     var jan1 = new Date(this.getFullYear(), 0, 1);
1539     var then = (7 - jan1.getDay() + 4);
1540     return String.leftPad(((now - then) / 7) + 1, 2, "0");
1541 };
1542
1543 /**
1544  * Whether or not the current date is in a leap year.
1545  * @return {Boolean} True if the current date is in a leap year, else false
1546  */
1547 Date.prototype.isLeapYear = function() {
1548     var year = this.getFullYear();
1549     return ((year & 3) == 0 && (year % 100 || (year % 400 == 0 && year)));
1550 };
1551
1552 /**
1553  * Get the first day of the current month, adjusted for leap year.  The returned value
1554  * is the numeric day index within the week (0-6) which can be used in conjunction with
1555  * the {@link #monthNames} array to retrieve the textual day name.
1556  * Example:
1557  *<pre><code>
1558 var dt = new Date('1/10/2007');
1559 document.write(Date.dayNames[dt.getFirstDayOfMonth()]); //output: 'Monday'
1560 </code></pre>
1561  * @return {Number} The day number (0-6)
1562  */
1563 Date.prototype.getFirstDayOfMonth = function() {
1564     var day = (this.getDay() - (this.getDate() - 1)) % 7;
1565     return (day < 0) ? (day + 7) : day;
1566 };
1567
1568 /**
1569  * Get the last day of the current month, adjusted for leap year.  The returned value
1570  * is the numeric day index within the week (0-6) which can be used in conjunction with
1571  * the {@link #monthNames} array to retrieve the textual day name.
1572  * Example:
1573  *<pre><code>
1574 var dt = new Date('1/10/2007');
1575 document.write(Date.dayNames[dt.getLastDayOfMonth()]); //output: 'Wednesday'
1576 </code></pre>
1577  * @return {Number} The day number (0-6)
1578  */
1579 Date.prototype.getLastDayOfMonth = function() {
1580     var day = (this.getDay() + (Date.daysInMonth[this.getMonth()] - this.getDate())) % 7;
1581     return (day < 0) ? (day + 7) : day;
1582 };
1583
1584
1585 /**
1586  * Get the first date of this date's month
1587  * @return {Date}
1588  */
1589 Date.prototype.getFirstDateOfMonth = function() {
1590     return new Date(this.getFullYear(), this.getMonth(), 1);
1591 };
1592
1593 /**
1594  * Get the last date of this date's month
1595  * @return {Date}
1596  */
1597 Date.prototype.getLastDateOfMonth = function() {
1598     return new Date(this.getFullYear(), this.getMonth(), this.getDaysInMonth());
1599 };
1600 /**
1601  * Get the number of days in the current month, adjusted for leap year.
1602  * @return {Number} The number of days in the month
1603  */
1604 Date.prototype.getDaysInMonth = function() {
1605     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1606     return Date.daysInMonth[this.getMonth()];
1607 };
1608
1609 /**
1610  * Get the English ordinal suffix of the current day (equivalent to the format specifier 'S').
1611  * @return {String} 'st, 'nd', 'rd' or 'th'
1612  */
1613 Date.prototype.getSuffix = function() {
1614     switch (this.getDate()) {
1615         case 1:
1616         case 21:
1617         case 31:
1618             return "st";
1619         case 2:
1620         case 22:
1621             return "nd";
1622         case 3:
1623         case 23:
1624             return "rd";
1625         default:
1626             return "th";
1627     }
1628 };
1629
1630 // private
1631 Date.daysInMonth = [31,28,31,30,31,30,31,31,30,31,30,31];
1632
1633 /**
1634  * An array of textual month names.
1635  * Override these values for international dates, for example...
1636  * Date.monthNames = ['JanInYourLang', 'FebInYourLang', ...];
1637  * @type Array
1638  * @static
1639  */
1640 Date.monthNames =
1641    ["January",
1642     "February",
1643     "March",
1644     "April",
1645     "May",
1646     "June",
1647     "July",
1648     "August",
1649     "September",
1650     "October",
1651     "November",
1652     "December"];
1653
1654 /**
1655  * An array of textual day names.
1656  * Override these values for international dates, for example...
1657  * Date.dayNames = ['SundayInYourLang', 'MondayInYourLang', ...];
1658  * @type Array
1659  * @static
1660  */
1661 Date.dayNames =
1662    ["Sunday",
1663     "Monday",
1664     "Tuesday",
1665     "Wednesday",
1666     "Thursday",
1667     "Friday",
1668     "Saturday"];
1669
1670 // private
1671 Date.y2kYear = 50;
1672 // private
1673 Date.monthNumbers = {
1674     Jan:0,
1675     Feb:1,
1676     Mar:2,
1677     Apr:3,
1678     May:4,
1679     Jun:5,
1680     Jul:6,
1681     Aug:7,
1682     Sep:8,
1683     Oct:9,
1684     Nov:10,
1685     Dec:11};
1686
1687 /**
1688  * Creates and returns a new Date instance with the exact same date value as the called instance.
1689  * Dates are copied and passed by reference, so if a copied date variable is modified later, the original
1690  * variable will also be changed.  When the intention is to create a new variable that will not
1691  * modify the original instance, you should create a clone.
1692  *
1693  * Example of correctly cloning a date:
1694  * <pre><code>
1695 //wrong way:
1696 var orig = new Date('10/1/2006');
1697 var copy = orig;
1698 copy.setDate(5);
1699 document.write(orig);  //returns 'Thu Oct 05 2006'!
1700
1701 //correct way:
1702 var orig = new Date('10/1/2006');
1703 var copy = orig.clone();
1704 copy.setDate(5);
1705 document.write(orig);  //returns 'Thu Oct 01 2006'
1706 </code></pre>
1707  * @return {Date} The new Date instance
1708  */
1709 Date.prototype.clone = function() {
1710         return new Date(this.getTime());
1711 };
1712
1713 /**
1714  * Clears any time information from this date
1715  @param {Boolean} clone true to create a clone of this date, clear the time and return it
1716  @return {Date} this or the clone
1717  */
1718 Date.prototype.clearTime = function(clone){
1719     if(clone){
1720         return this.clone().clearTime();
1721     }
1722     this.setHours(0);
1723     this.setMinutes(0);
1724     this.setSeconds(0);
1725     this.setMilliseconds(0);
1726     return this;
1727 };
1728
1729 // private
1730 // safari setMonth is broken -- check that this is only donw once...
1731 if(Roo.isSafari && typeof(Date.brokenSetMonth) == 'undefined'){
1732     Date.brokenSetMonth = Date.prototype.setMonth;
1733         Date.prototype.setMonth = function(num){
1734                 if(num <= -1){
1735                         var n = Math.ceil(-num);
1736                         var back_year = Math.ceil(n/12);
1737                         var month = (n % 12) ? 12 - n % 12 : 0 ;
1738                         this.setFullYear(this.getFullYear() - back_year);
1739                         return Date.brokenSetMonth.call(this, month);
1740                 } else {
1741                         return Date.brokenSetMonth.apply(this, arguments);
1742                 }
1743         };
1744 }
1745
1746 /** Date interval constant 
1747 * @static 
1748 * @type String */
1749 Date.MILLI = "ms";
1750 /** Date interval constant 
1751 * @static 
1752 * @type String */
1753 Date.SECOND = "s";
1754 /** Date interval constant 
1755 * @static 
1756 * @type String */
1757 Date.MINUTE = "mi";
1758 /** Date interval constant 
1759 * @static 
1760 * @type String */
1761 Date.HOUR = "h";
1762 /** Date interval constant 
1763 * @static 
1764 * @type String */
1765 Date.DAY = "d";
1766 /** Date interval constant 
1767 * @static 
1768 * @type String */
1769 Date.MONTH = "mo";
1770 /** Date interval constant 
1771 * @static 
1772 * @type String */
1773 Date.YEAR = "y";
1774
1775 /**
1776  * Provides a convenient method of performing basic date arithmetic.  This method
1777  * does not modify the Date instance being called - it creates and returns
1778  * a new Date instance containing the resulting date value.
1779  *
1780  * Examples:
1781  * <pre><code>
1782 //Basic usage:
1783 var dt = new Date('10/29/2006').add(Date.DAY, 5);
1784 document.write(dt); //returns 'Fri Oct 06 2006 00:00:00'
1785
1786 //Negative values will subtract correctly:
1787 var dt2 = new Date('10/1/2006').add(Date.DAY, -5);
1788 document.write(dt2); //returns 'Tue Sep 26 2006 00:00:00'
1789
1790 //You can even chain several calls together in one line!
1791 var dt3 = new Date('10/1/2006').add(Date.DAY, 5).add(Date.HOUR, 8).add(Date.MINUTE, -30);
1792 document.write(dt3); //returns 'Fri Oct 06 2006 07:30:00'
1793  </code></pre>
1794  *
1795  * @param {String} interval   A valid date interval enum value
1796  * @param {Number} value      The amount to add to the current date
1797  * @return {Date} The new Date instance
1798  */
1799 Date.prototype.add = function(interval, value){
1800   var d = this.clone();
1801   if (!interval || value === 0) { return d; }
1802   switch(interval.toLowerCase()){
1803     case Date.MILLI:
1804       d.setMilliseconds(this.getMilliseconds() + value);
1805       break;
1806     case Date.SECOND:
1807       d.setSeconds(this.getSeconds() + value);
1808       break;
1809     case Date.MINUTE:
1810       d.setMinutes(this.getMinutes() + value);
1811       break;
1812     case Date.HOUR:
1813       d.setHours(this.getHours() + value);
1814       break;
1815     case Date.DAY:
1816       d.setDate(this.getDate() + value);
1817       break;
1818     case Date.MONTH:
1819       var day = this.getDate();
1820       if(day > 28){
1821           day = Math.min(day, this.getFirstDateOfMonth().add('mo', value).getLastDateOfMonth().getDate());
1822       }
1823       d.setDate(day);
1824       d.setMonth(this.getMonth() + value);
1825       break;
1826     case Date.YEAR:
1827       d.setFullYear(this.getFullYear() + value);
1828       break;
1829   }
1830   return d;
1831 };
1832 /*
1833  * Based on:
1834  * Ext JS Library 1.1.1
1835  * Copyright(c) 2006-2007, Ext JS, LLC.
1836  *
1837  * Originally Released Under LGPL - original licence link has changed is not relivant.
1838  *
1839  * Fork - LGPL
1840  * <script type="text/javascript">
1841  */
1842
1843 /**
1844  * @class Roo.lib.Dom
1845  * @static
1846  * 
1847  * Dom utils (from YIU afaik)
1848  * 
1849  **/
1850 Roo.lib.Dom = {
1851     /**
1852      * Get the view width
1853      * @param {Boolean} full True will get the full document, otherwise it's the view width
1854      * @return {Number} The width
1855      */
1856      
1857     getViewWidth : function(full) {
1858         return full ? this.getDocumentWidth() : this.getViewportWidth();
1859     },
1860     /**
1861      * Get the view height
1862      * @param {Boolean} full True will get the full document, otherwise it's the view height
1863      * @return {Number} The height
1864      */
1865     getViewHeight : function(full) {
1866         return full ? this.getDocumentHeight() : this.getViewportHeight();
1867     },
1868
1869     getDocumentHeight: function() {
1870         var scrollHeight = (document.compatMode != "CSS1Compat") ? document.body.scrollHeight : document.documentElement.scrollHeight;
1871         return Math.max(scrollHeight, this.getViewportHeight());
1872     },
1873
1874     getDocumentWidth: function() {
1875         var scrollWidth = (document.compatMode != "CSS1Compat") ? document.body.scrollWidth : document.documentElement.scrollWidth;
1876         return Math.max(scrollWidth, this.getViewportWidth());
1877     },
1878
1879     getViewportHeight: function() {
1880         var height = self.innerHeight;
1881         var mode = document.compatMode;
1882
1883         if ((mode || Roo.isIE) && !Roo.isOpera) {
1884             height = (mode == "CSS1Compat") ?
1885                      document.documentElement.clientHeight :
1886                      document.body.clientHeight;
1887         }
1888
1889         return height;
1890     },
1891
1892     getViewportWidth: function() {
1893         var width = self.innerWidth;
1894         var mode = document.compatMode;
1895
1896         if (mode || Roo.isIE) {
1897             width = (mode == "CSS1Compat") ?
1898                     document.documentElement.clientWidth :
1899                     document.body.clientWidth;
1900         }
1901         return width;
1902     },
1903
1904     isAncestor : function(p, c) {
1905         p = Roo.getDom(p);
1906         c = Roo.getDom(c);
1907         if (!p || !c) {
1908             return false;
1909         }
1910
1911         if (p.contains && !Roo.isSafari) {
1912             return p.contains(c);
1913         } else if (p.compareDocumentPosition) {
1914             return !!(p.compareDocumentPosition(c) & 16);
1915         } else {
1916             var parent = c.parentNode;
1917             while (parent) {
1918                 if (parent == p) {
1919                     return true;
1920                 }
1921                 else if (!parent.tagName || parent.tagName.toUpperCase() == "HTML") {
1922                     return false;
1923                 }
1924                 parent = parent.parentNode;
1925             }
1926             return false;
1927         }
1928     },
1929
1930     getRegion : function(el) {
1931         return Roo.lib.Region.getRegion(el);
1932     },
1933
1934     getY : function(el) {
1935         return this.getXY(el)[1];
1936     },
1937
1938     getX : function(el) {
1939         return this.getXY(el)[0];
1940     },
1941
1942     getXY : function(el) {
1943         var p, pe, b, scroll, bd = document.body;
1944         el = Roo.getDom(el);
1945         var fly = Roo.lib.AnimBase.fly;
1946         if (el.getBoundingClientRect) {
1947             b = el.getBoundingClientRect();
1948             scroll = fly(document).getScroll();
1949             return [b.left + scroll.left, b.top + scroll.top];
1950         }
1951         var x = 0, y = 0;
1952
1953         p = el;
1954
1955         var hasAbsolute = fly(el).getStyle("position") == "absolute";
1956
1957         while (p) {
1958
1959             x += p.offsetLeft;
1960             y += p.offsetTop;
1961
1962             if (!hasAbsolute && fly(p).getStyle("position") == "absolute") {
1963                 hasAbsolute = true;
1964             }
1965
1966             if (Roo.isGecko) {
1967                 pe = fly(p);
1968
1969                 var bt = parseInt(pe.getStyle("borderTopWidth"), 10) || 0;
1970                 var bl = parseInt(pe.getStyle("borderLeftWidth"), 10) || 0;
1971
1972
1973                 x += bl;
1974                 y += bt;
1975
1976
1977                 if (p != el && pe.getStyle('overflow') != 'visible') {
1978                     x += bl;
1979                     y += bt;
1980                 }
1981             }
1982             p = p.offsetParent;
1983         }
1984
1985         if (Roo.isSafari && hasAbsolute) {
1986             x -= bd.offsetLeft;
1987             y -= bd.offsetTop;
1988         }
1989
1990         if (Roo.isGecko && !hasAbsolute) {
1991             var dbd = fly(bd);
1992             x += parseInt(dbd.getStyle("borderLeftWidth"), 10) || 0;
1993             y += parseInt(dbd.getStyle("borderTopWidth"), 10) || 0;
1994         }
1995
1996         p = el.parentNode;
1997         while (p && p != bd) {
1998             if (!Roo.isOpera || (p.tagName != 'TR' && fly(p).getStyle("display") != "inline")) {
1999                 x -= p.scrollLeft;
2000                 y -= p.scrollTop;
2001             }
2002             p = p.parentNode;
2003         }
2004         return [x, y];
2005     },
2006  
2007   
2008
2009
2010     setXY : function(el, xy) {
2011         el = Roo.fly(el, '_setXY');
2012         el.position();
2013         var pts = el.translatePoints(xy);
2014         if (xy[0] !== false) {
2015             el.dom.style.left = pts.left + "px";
2016         }
2017         if (xy[1] !== false) {
2018             el.dom.style.top = pts.top + "px";
2019         }
2020     },
2021
2022     setX : function(el, x) {
2023         this.setXY(el, [x, false]);
2024     },
2025
2026     setY : function(el, y) {
2027         this.setXY(el, [false, y]);
2028     }
2029 };
2030 /*
2031  * Portions of this file are based on pieces of Yahoo User Interface Library
2032  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2033  * YUI licensed under the BSD License:
2034  * http://developer.yahoo.net/yui/license.txt
2035  * <script type="text/javascript">
2036  *
2037  */
2038
2039 Roo.lib.Event = function() {
2040     var loadComplete = false;
2041     var listeners = [];
2042     var unloadListeners = [];
2043     var retryCount = 0;
2044     var onAvailStack = [];
2045     var counter = 0;
2046     var lastError = null;
2047
2048     return {
2049         POLL_RETRYS: 200,
2050         POLL_INTERVAL: 20,
2051         EL: 0,
2052         TYPE: 1,
2053         FN: 2,
2054         WFN: 3,
2055         OBJ: 3,
2056         ADJ_SCOPE: 4,
2057         _interval: null,
2058
2059         startInterval: function() {
2060             if (!this._interval) {
2061                 var self = this;
2062                 var callback = function() {
2063                     self._tryPreloadAttach();
2064                 };
2065                 this._interval = setInterval(callback, this.POLL_INTERVAL);
2066
2067             }
2068         },
2069
2070         onAvailable: function(p_id, p_fn, p_obj, p_override) {
2071             onAvailStack.push({ id:         p_id,
2072                 fn:         p_fn,
2073                 obj:        p_obj,
2074                 override:   p_override,
2075                 checkReady: false    });
2076
2077             retryCount = this.POLL_RETRYS;
2078             this.startInterval();
2079         },
2080
2081
2082         addListener: function(el, eventName, fn) {
2083             el = Roo.getDom(el);
2084             if (!el || !fn) {
2085                 return false;
2086             }
2087
2088             if ("unload" == eventName) {
2089                 unloadListeners[unloadListeners.length] =
2090                 [el, eventName, fn];
2091                 return true;
2092             }
2093
2094             var wrappedFn = function(e) {
2095                 return fn(Roo.lib.Event.getEvent(e));
2096             };
2097
2098             var li = [el, eventName, fn, wrappedFn];
2099
2100             var index = listeners.length;
2101             listeners[index] = li;
2102
2103             this.doAdd(el, eventName, wrappedFn, false);
2104             return true;
2105
2106         },
2107
2108
2109         removeListener: function(el, eventName, fn) {
2110             var i, len;
2111
2112             el = Roo.getDom(el);
2113
2114             if(!fn) {
2115                 return this.purgeElement(el, false, eventName);
2116             }
2117
2118
2119             if ("unload" == eventName) {
2120
2121                 for (i = 0,len = unloadListeners.length; i < len; i++) {
2122                     var li = unloadListeners[i];
2123                     if (li &&
2124                         li[0] == el &&
2125                         li[1] == eventName &&
2126                         li[2] == fn) {
2127                         unloadListeners.splice(i, 1);
2128                         return true;
2129                     }
2130                 }
2131
2132                 return false;
2133             }
2134
2135             var cacheItem = null;
2136
2137
2138             var index = arguments[3];
2139
2140             if ("undefined" == typeof index) {
2141                 index = this._getCacheIndex(el, eventName, fn);
2142             }
2143
2144             if (index >= 0) {
2145                 cacheItem = listeners[index];
2146             }
2147
2148             if (!el || !cacheItem) {
2149                 return false;
2150             }
2151
2152             this.doRemove(el, eventName, cacheItem[this.WFN], false);
2153
2154             delete listeners[index][this.WFN];
2155             delete listeners[index][this.FN];
2156             listeners.splice(index, 1);
2157
2158             return true;
2159
2160         },
2161
2162
2163         getTarget: function(ev, resolveTextNode) {
2164             ev = ev.browserEvent || ev;
2165             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2166             var t = ev.target || ev.srcElement;
2167             return this.resolveTextNode(t);
2168         },
2169
2170
2171         resolveTextNode: function(node) {
2172             if (Roo.isSafari && node && 3 == node.nodeType) {
2173                 return node.parentNode;
2174             } else {
2175                 return node;
2176             }
2177         },
2178
2179
2180         getPageX: function(ev) {
2181             ev = ev.browserEvent || ev;
2182             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2183             var x = ev.pageX;
2184             if (!x && 0 !== x) {
2185                 x = ev.clientX || 0;
2186
2187                 if (Roo.isIE) {
2188                     x += this.getScroll()[1];
2189                 }
2190             }
2191
2192             return x;
2193         },
2194
2195
2196         getPageY: function(ev) {
2197             ev = ev.browserEvent || ev;
2198             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2199             var y = ev.pageY;
2200             if (!y && 0 !== y) {
2201                 y = ev.clientY || 0;
2202
2203                 if (Roo.isIE) {
2204                     y += this.getScroll()[0];
2205                 }
2206             }
2207
2208
2209             return y;
2210         },
2211
2212
2213         getXY: function(ev) {
2214             ev = ev.browserEvent || ev;
2215             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2216             return [this.getPageX(ev), this.getPageY(ev)];
2217         },
2218
2219
2220         getRelatedTarget: function(ev) {
2221             ev = ev.browserEvent || ev;
2222             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2223             var t = ev.relatedTarget;
2224             if (!t) {
2225                 if (ev.type == "mouseout") {
2226                     t = ev.toElement;
2227                 } else if (ev.type == "mouseover") {
2228                     t = ev.fromElement;
2229                 }
2230             }
2231
2232             return this.resolveTextNode(t);
2233         },
2234
2235
2236         getTime: function(ev) {
2237             ev = ev.browserEvent || ev;
2238             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2239             if (!ev.time) {
2240                 var t = new Date().getTime();
2241                 try {
2242                     ev.time = t;
2243                 } catch(ex) {
2244                     this.lastError = ex;
2245                     return t;
2246                 }
2247             }
2248
2249             return ev.time;
2250         },
2251
2252
2253         stopEvent: function(ev) {
2254             this.stopPropagation(ev);
2255             this.preventDefault(ev);
2256         },
2257
2258
2259         stopPropagation: function(ev) {
2260             ev = ev.browserEvent || ev;
2261             if (ev.stopPropagation) {
2262                 ev.stopPropagation();
2263             } else {
2264                 ev.cancelBubble = true;
2265             }
2266         },
2267
2268
2269         preventDefault: function(ev) {
2270             ev = ev.browserEvent || ev;
2271             if(ev.preventDefault) {
2272                 ev.preventDefault();
2273             } else {
2274                 ev.returnValue = false;
2275             }
2276         },
2277
2278
2279         getEvent: function(e) {
2280             var ev = e || window.event;
2281             if (!ev) {
2282                 var c = this.getEvent.caller;
2283                 while (c) {
2284                     ev = c.arguments[0];
2285                     if (ev && Event == ev.constructor) {
2286                         break;
2287                     }
2288                     c = c.caller;
2289                 }
2290             }
2291             return ev;
2292         },
2293
2294
2295         getCharCode: function(ev) {
2296             ev = ev.browserEvent || ev;
2297             return ev.charCode || ev.keyCode || 0;
2298         },
2299
2300
2301         _getCacheIndex: function(el, eventName, fn) {
2302             for (var i = 0,len = listeners.length; i < len; ++i) {
2303                 var li = listeners[i];
2304                 if (li &&
2305                     li[this.FN] == fn &&
2306                     li[this.EL] == el &&
2307                     li[this.TYPE] == eventName) {
2308                     return i;
2309                 }
2310             }
2311
2312             return -1;
2313         },
2314
2315
2316         elCache: {},
2317
2318
2319         getEl: function(id) {
2320             return document.getElementById(id);
2321         },
2322
2323
2324         clearCache: function() {
2325         },
2326
2327
2328         _load: function(e) {
2329             loadComplete = true;
2330             var EU = Roo.lib.Event;
2331
2332
2333             if (Roo.isIE) {
2334                 EU.doRemove(window, "load", EU._load);
2335             }
2336         },
2337
2338
2339         _tryPreloadAttach: function() {
2340
2341             if (this.locked) {
2342                 return false;
2343             }
2344
2345             this.locked = true;
2346
2347
2348             var tryAgain = !loadComplete;
2349             if (!tryAgain) {
2350                 tryAgain = (retryCount > 0);
2351             }
2352
2353
2354             var notAvail = [];
2355             for (var i = 0,len = onAvailStack.length; i < len; ++i) {
2356                 var item = onAvailStack[i];
2357                 if (item) {
2358                     var el = this.getEl(item.id);
2359
2360                     if (el) {
2361                         if (!item.checkReady ||
2362                             loadComplete ||
2363                             el.nextSibling ||
2364                             (document && document.body)) {
2365
2366                             var scope = el;
2367                             if (item.override) {
2368                                 if (item.override === true) {
2369                                     scope = item.obj;
2370                                 } else {
2371                                     scope = item.override;
2372                                 }
2373                             }
2374                             item.fn.call(scope, item.obj);
2375                             onAvailStack[i] = null;
2376                         }
2377                     } else {
2378                         notAvail.push(item);
2379                     }
2380                 }
2381             }
2382
2383             retryCount = (notAvail.length === 0) ? 0 : retryCount - 1;
2384
2385             if (tryAgain) {
2386
2387                 this.startInterval();
2388             } else {
2389                 clearInterval(this._interval);
2390                 this._interval = null;
2391             }
2392
2393             this.locked = false;
2394
2395             return true;
2396
2397         },
2398
2399
2400         purgeElement: function(el, recurse, eventName) {
2401             var elListeners = this.getListeners(el, eventName);
2402             if (elListeners) {
2403                 for (var i = 0,len = elListeners.length; i < len; ++i) {
2404                     var l = elListeners[i];
2405                     this.removeListener(el, l.type, l.fn);
2406                 }
2407             }
2408
2409             if (recurse && el && el.childNodes) {
2410                 for (i = 0,len = el.childNodes.length; i < len; ++i) {
2411                     this.purgeElement(el.childNodes[i], recurse, eventName);
2412                 }
2413             }
2414         },
2415
2416
2417         getListeners: function(el, eventName) {
2418             var results = [], searchLists;
2419             if (!eventName) {
2420                 searchLists = [listeners, unloadListeners];
2421             } else if (eventName == "unload") {
2422                 searchLists = [unloadListeners];
2423             } else {
2424                 searchLists = [listeners];
2425             }
2426
2427             for (var j = 0; j < searchLists.length; ++j) {
2428                 var searchList = searchLists[j];
2429                 if (searchList && searchList.length > 0) {
2430                     for (var i = 0,len = searchList.length; i < len; ++i) {
2431                         var l = searchList[i];
2432                         if (l && l[this.EL] === el &&
2433                             (!eventName || eventName === l[this.TYPE])) {
2434                             results.push({
2435                                 type:   l[this.TYPE],
2436                                 fn:     l[this.FN],
2437                                 obj:    l[this.OBJ],
2438                                 adjust: l[this.ADJ_SCOPE],
2439                                 index:  i
2440                             });
2441                         }
2442                     }
2443                 }
2444             }
2445
2446             return (results.length) ? results : null;
2447         },
2448
2449
2450         _unload: function(e) {
2451
2452             var EU = Roo.lib.Event, i, j, l, len, index;
2453
2454             for (i = 0,len = unloadListeners.length; i < len; ++i) {
2455                 l = unloadListeners[i];
2456                 if (l) {
2457                     var scope = window;
2458                     if (l[EU.ADJ_SCOPE]) {
2459                         if (l[EU.ADJ_SCOPE] === true) {
2460                             scope = l[EU.OBJ];
2461                         } else {
2462                             scope = l[EU.ADJ_SCOPE];
2463                         }
2464                     }
2465                     l[EU.FN].call(scope, EU.getEvent(e), l[EU.OBJ]);
2466                     unloadListeners[i] = null;
2467                     l = null;
2468                     scope = null;
2469                 }
2470             }
2471
2472             unloadListeners = null;
2473
2474             if (listeners && listeners.length > 0) {
2475                 j = listeners.length;
2476                 while (j) {
2477                     index = j - 1;
2478                     l = listeners[index];
2479                     if (l) {
2480                         EU.removeListener(l[EU.EL], l[EU.TYPE],
2481                                 l[EU.FN], index);
2482                     }
2483                     j = j - 1;
2484                 }
2485                 l = null;
2486
2487                 EU.clearCache();
2488             }
2489
2490             EU.doRemove(window, "unload", EU._unload);
2491
2492         },
2493
2494
2495         getScroll: function() {
2496             var dd = document.documentElement, db = document.body;
2497             if (dd && (dd.scrollTop || dd.scrollLeft)) {
2498                 return [dd.scrollTop, dd.scrollLeft];
2499             } else if (db) {
2500                 return [db.scrollTop, db.scrollLeft];
2501             } else {
2502                 return [0, 0];
2503             }
2504         },
2505
2506
2507         doAdd: function () {
2508             if (window.addEventListener) {
2509                 return function(el, eventName, fn, capture) {
2510                     el.addEventListener(eventName, fn, (capture));
2511                 };
2512             } else if (window.attachEvent) {
2513                 return function(el, eventName, fn, capture) {
2514                     el.attachEvent("on" + eventName, fn);
2515                 };
2516             } else {
2517                 return function() {
2518                 };
2519             }
2520         }(),
2521
2522
2523         doRemove: function() {
2524             if (window.removeEventListener) {
2525                 return function (el, eventName, fn, capture) {
2526                     el.removeEventListener(eventName, fn, (capture));
2527                 };
2528             } else if (window.detachEvent) {
2529                 return function (el, eventName, fn) {
2530                     el.detachEvent("on" + eventName, fn);
2531                 };
2532             } else {
2533                 return function() {
2534                 };
2535             }
2536         }()
2537     };
2538     
2539 }();
2540 (function() {     
2541    
2542     var E = Roo.lib.Event;
2543     E.on = E.addListener;
2544     E.un = E.removeListener;
2545
2546     if (document && document.body) {
2547         E._load();
2548     } else {
2549         E.doAdd(window, "load", E._load);
2550     }
2551     E.doAdd(window, "unload", E._unload);
2552     E._tryPreloadAttach();
2553 })();
2554
2555 /*
2556  * Portions of this file are based on pieces of Yahoo User Interface Library
2557  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2558  * YUI licensed under the BSD License:
2559  * http://developer.yahoo.net/yui/license.txt
2560  * <script type="text/javascript">
2561  *
2562  */
2563
2564 (function() {
2565     /**
2566      * @class Roo.lib.Ajax
2567      *
2568      */
2569     Roo.lib.Ajax = {
2570         /**
2571          * @static 
2572          */
2573         request : function(method, uri, cb, data, options) {
2574             if(options){
2575                 var hs = options.headers;
2576                 if(hs){
2577                     for(var h in hs){
2578                         if(hs.hasOwnProperty(h)){
2579                             this.initHeader(h, hs[h], false);
2580                         }
2581                     }
2582                 }
2583                 if(options.xmlData){
2584                     this.initHeader('Content-Type', 'text/xml', false);
2585                     method = 'POST';
2586                     data = options.xmlData;
2587                 }
2588             }
2589
2590             return this.asyncRequest(method, uri, cb, data);
2591         },
2592
2593         serializeForm : function(form) {
2594             if(typeof form == 'string') {
2595                 form = (document.getElementById(form) || document.forms[form]);
2596             }
2597
2598             var el, name, val, disabled, data = '', hasSubmit = false;
2599             for (var i = 0; i < form.elements.length; i++) {
2600                 el = form.elements[i];
2601                 disabled = form.elements[i].disabled;
2602                 name = form.elements[i].name;
2603                 val = form.elements[i].value;
2604
2605                 if (!disabled && name){
2606                     switch (el.type)
2607                             {
2608                         case 'select-one':
2609                         case 'select-multiple':
2610                             for (var j = 0; j < el.options.length; j++) {
2611                                 if (el.options[j].selected) {
2612                                     if (Roo.isIE) {
2613                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].attributes['value'].specified ? el.options[j].value : el.options[j].text) + '&';
2614                                     }
2615                                     else {
2616                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].hasAttribute('value') ? el.options[j].value : el.options[j].text) + '&';
2617                                     }
2618                                 }
2619                             }
2620                             break;
2621                         case 'radio':
2622                         case 'checkbox':
2623                             if (el.checked) {
2624                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2625                             }
2626                             break;
2627                         case 'file':
2628
2629                         case undefined:
2630
2631                         case 'reset':
2632
2633                         case 'button':
2634
2635                             break;
2636                         case 'submit':
2637                             if(hasSubmit == false) {
2638                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2639                                 hasSubmit = true;
2640                             }
2641                             break;
2642                         default:
2643                             data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2644                             break;
2645                     }
2646                 }
2647             }
2648             data = data.substr(0, data.length - 1);
2649             return data;
2650         },
2651
2652         headers:{},
2653
2654         hasHeaders:false,
2655
2656         useDefaultHeader:true,
2657
2658         defaultPostHeader:'application/x-www-form-urlencoded',
2659
2660         useDefaultXhrHeader:true,
2661
2662         defaultXhrHeader:'XMLHttpRequest',
2663
2664         hasDefaultHeaders:true,
2665
2666         defaultHeaders:{},
2667
2668         poll:{},
2669
2670         timeout:{},
2671
2672         pollInterval:50,
2673
2674         transactionId:0,
2675
2676         setProgId:function(id)
2677         {
2678             this.activeX.unshift(id);
2679         },
2680
2681         setDefaultPostHeader:function(b)
2682         {
2683             this.useDefaultHeader = b;
2684         },
2685
2686         setDefaultXhrHeader:function(b)
2687         {
2688             this.useDefaultXhrHeader = b;
2689         },
2690
2691         setPollingInterval:function(i)
2692         {
2693             if (typeof i == 'number' && isFinite(i)) {
2694                 this.pollInterval = i;
2695             }
2696         },
2697
2698         createXhrObject:function(transactionId)
2699         {
2700             var obj,http;
2701             try
2702             {
2703
2704                 http = new XMLHttpRequest();
2705
2706                 obj = { conn:http, tId:transactionId };
2707             }
2708             catch(e)
2709             {
2710                 for (var i = 0; i < this.activeX.length; ++i) {
2711                     try
2712                     {
2713
2714                         http = new ActiveXObject(this.activeX[i]);
2715
2716                         obj = { conn:http, tId:transactionId };
2717                         break;
2718                     }
2719                     catch(e) {
2720                     }
2721                 }
2722             }
2723             finally
2724             {
2725                 return obj;
2726             }
2727         },
2728
2729         getConnectionObject:function()
2730         {
2731             var o;
2732             var tId = this.transactionId;
2733
2734             try
2735             {
2736                 o = this.createXhrObject(tId);
2737                 if (o) {
2738                     this.transactionId++;
2739                 }
2740             }
2741             catch(e) {
2742             }
2743             finally
2744             {
2745                 return o;
2746             }
2747         },
2748
2749         asyncRequest:function(method, uri, callback, postData)
2750         {
2751             var o = this.getConnectionObject();
2752
2753             if (!o) {
2754                 return null;
2755             }
2756             else {
2757                 o.conn.open(method, uri, true);
2758
2759                 if (this.useDefaultXhrHeader) {
2760                     if (!this.defaultHeaders['X-Requested-With']) {
2761                         this.initHeader('X-Requested-With', this.defaultXhrHeader, true);
2762                     }
2763                 }
2764
2765                 if(postData && this.useDefaultHeader){
2766                     this.initHeader('Content-Type', this.defaultPostHeader);
2767                 }
2768
2769                  if (this.hasDefaultHeaders || this.hasHeaders) {
2770                     this.setHeader(o);
2771                 }
2772
2773                 this.handleReadyState(o, callback);
2774                 o.conn.send(postData || null);
2775
2776                 return o;
2777             }
2778         },
2779
2780         handleReadyState:function(o, callback)
2781         {
2782             var oConn = this;
2783
2784             if (callback && callback.timeout) {
2785                 
2786                 this.timeout[o.tId] = window.setTimeout(function() {
2787                     oConn.abort(o, callback, true);
2788                 }, callback.timeout);
2789             }
2790
2791             this.poll[o.tId] = window.setInterval(
2792                     function() {
2793                         if (o.conn && o.conn.readyState == 4) {
2794                             window.clearInterval(oConn.poll[o.tId]);
2795                             delete oConn.poll[o.tId];
2796
2797                             if(callback && callback.timeout) {
2798                                 window.clearTimeout(oConn.timeout[o.tId]);
2799                                 delete oConn.timeout[o.tId];
2800                             }
2801
2802                             oConn.handleTransactionResponse(o, callback);
2803                         }
2804                     }
2805                     , this.pollInterval);
2806         },
2807
2808         handleTransactionResponse:function(o, callback, isAbort)
2809         {
2810
2811             if (!callback) {
2812                 this.releaseObject(o);
2813                 return;
2814             }
2815
2816             var httpStatus, responseObject;
2817
2818             try
2819             {
2820                 if (o.conn.status !== undefined && o.conn.status != 0) {
2821                     httpStatus = o.conn.status;
2822                 }
2823                 else {
2824                     httpStatus = 13030;
2825                 }
2826             }
2827             catch(e) {
2828
2829
2830                 httpStatus = 13030;
2831             }
2832
2833             if (httpStatus >= 200 && httpStatus < 300) {
2834                 responseObject = this.createResponseObject(o, callback.argument);
2835                 if (callback.success) {
2836                     if (!callback.scope) {
2837                         callback.success(responseObject);
2838                     }
2839                     else {
2840
2841
2842                         callback.success.apply(callback.scope, [responseObject]);
2843                     }
2844                 }
2845             }
2846             else {
2847                 switch (httpStatus) {
2848
2849                     case 12002:
2850                     case 12029:
2851                     case 12030:
2852                     case 12031:
2853                     case 12152:
2854                     case 13030:
2855                         responseObject = this.createExceptionObject(o.tId, callback.argument, (isAbort ? isAbort : false));
2856                         if (callback.failure) {
2857                             if (!callback.scope) {
2858                                 callback.failure(responseObject);
2859                             }
2860                             else {
2861                                 callback.failure.apply(callback.scope, [responseObject]);
2862                             }
2863                         }
2864                         break;
2865                     default:
2866                         responseObject = this.createResponseObject(o, callback.argument);
2867                         if (callback.failure) {
2868                             if (!callback.scope) {
2869                                 callback.failure(responseObject);
2870                             }
2871                             else {
2872                                 callback.failure.apply(callback.scope, [responseObject]);
2873                             }
2874                         }
2875                 }
2876             }
2877
2878             this.releaseObject(o);
2879             responseObject = null;
2880         },
2881
2882         createResponseObject:function(o, callbackArg)
2883         {
2884             var obj = {};
2885             var headerObj = {};
2886
2887             try
2888             {
2889                 var headerStr = o.conn.getAllResponseHeaders();
2890                 var header = headerStr.split('\n');
2891                 for (var i = 0; i < header.length; i++) {
2892                     var delimitPos = header[i].indexOf(':');
2893                     if (delimitPos != -1) {
2894                         headerObj[header[i].substring(0, delimitPos)] = header[i].substring(delimitPos + 2);
2895                     }
2896                 }
2897             }
2898             catch(e) {
2899             }
2900
2901             obj.tId = o.tId;
2902             obj.status = o.conn.status;
2903             obj.statusText = o.conn.statusText;
2904             obj.getResponseHeader = headerObj;
2905             obj.getAllResponseHeaders = headerStr;
2906             obj.responseText = o.conn.responseText;
2907             obj.responseXML = o.conn.responseXML;
2908
2909             if (typeof callbackArg !== undefined) {
2910                 obj.argument = callbackArg;
2911             }
2912
2913             return obj;
2914         },
2915
2916         createExceptionObject:function(tId, callbackArg, isAbort)
2917         {
2918             var COMM_CODE = 0;
2919             var COMM_ERROR = 'communication failure';
2920             var ABORT_CODE = -1;
2921             var ABORT_ERROR = 'transaction aborted';
2922
2923             var obj = {};
2924
2925             obj.tId = tId;
2926             if (isAbort) {
2927                 obj.status = ABORT_CODE;
2928                 obj.statusText = ABORT_ERROR;
2929             }
2930             else {
2931                 obj.status = COMM_CODE;
2932                 obj.statusText = COMM_ERROR;
2933             }
2934
2935             if (callbackArg) {
2936                 obj.argument = callbackArg;
2937             }
2938
2939             return obj;
2940         },
2941
2942         initHeader:function(label, value, isDefault)
2943         {
2944             var headerObj = (isDefault) ? this.defaultHeaders : this.headers;
2945
2946             if (headerObj[label] === undefined) {
2947                 headerObj[label] = value;
2948             }
2949             else {
2950
2951
2952                 headerObj[label] = value + "," + headerObj[label];
2953             }
2954
2955             if (isDefault) {
2956                 this.hasDefaultHeaders = true;
2957             }
2958             else {
2959                 this.hasHeaders = true;
2960             }
2961         },
2962
2963
2964         setHeader:function(o)
2965         {
2966             if (this.hasDefaultHeaders) {
2967                 for (var prop in this.defaultHeaders) {
2968                     if (this.defaultHeaders.hasOwnProperty(prop)) {
2969                         o.conn.setRequestHeader(prop, this.defaultHeaders[prop]);
2970                     }
2971                 }
2972             }
2973
2974             if (this.hasHeaders) {
2975                 for (var prop in this.headers) {
2976                     if (this.headers.hasOwnProperty(prop)) {
2977                         o.conn.setRequestHeader(prop, this.headers[prop]);
2978                     }
2979                 }
2980                 this.headers = {};
2981                 this.hasHeaders = false;
2982             }
2983         },
2984
2985         resetDefaultHeaders:function() {
2986             delete this.defaultHeaders;
2987             this.defaultHeaders = {};
2988             this.hasDefaultHeaders = false;
2989         },
2990
2991         abort:function(o, callback, isTimeout)
2992         {
2993             if(this.isCallInProgress(o)) {
2994                 o.conn.abort();
2995                 window.clearInterval(this.poll[o.tId]);
2996                 delete this.poll[o.tId];
2997                 if (isTimeout) {
2998                     delete this.timeout[o.tId];
2999                 }
3000
3001                 this.handleTransactionResponse(o, callback, true);
3002
3003                 return true;
3004             }
3005             else {
3006                 return false;
3007             }
3008         },
3009
3010
3011         isCallInProgress:function(o)
3012         {
3013             if (o && o.conn) {
3014                 return o.conn.readyState != 4 && o.conn.readyState != 0;
3015             }
3016             else {
3017
3018                 return false;
3019             }
3020         },
3021
3022
3023         releaseObject:function(o)
3024         {
3025
3026             o.conn = null;
3027
3028             o = null;
3029         },
3030
3031         activeX:[
3032         'MSXML2.XMLHTTP.3.0',
3033         'MSXML2.XMLHTTP',
3034         'Microsoft.XMLHTTP'
3035         ]
3036
3037
3038     };
3039 })();/*
3040  * Portions of this file are based on pieces of Yahoo User Interface Library
3041  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3042  * YUI licensed under the BSD License:
3043  * http://developer.yahoo.net/yui/license.txt
3044  * <script type="text/javascript">
3045  *
3046  */
3047
3048 Roo.lib.Region = function(t, r, b, l) {
3049     this.top = t;
3050     this[1] = t;
3051     this.right = r;
3052     this.bottom = b;
3053     this.left = l;
3054     this[0] = l;
3055 };
3056
3057
3058 Roo.lib.Region.prototype = {
3059     contains : function(region) {
3060         return ( region.left >= this.left &&
3061                  region.right <= this.right &&
3062                  region.top >= this.top &&
3063                  region.bottom <= this.bottom    );
3064
3065     },
3066
3067     getArea : function() {
3068         return ( (this.bottom - this.top) * (this.right - this.left) );
3069     },
3070
3071     intersect : function(region) {
3072         var t = Math.max(this.top, region.top);
3073         var r = Math.min(this.right, region.right);
3074         var b = Math.min(this.bottom, region.bottom);
3075         var l = Math.max(this.left, region.left);
3076
3077         if (b >= t && r >= l) {
3078             return new Roo.lib.Region(t, r, b, l);
3079         } else {
3080             return null;
3081         }
3082     },
3083     union : function(region) {
3084         var t = Math.min(this.top, region.top);
3085         var r = Math.max(this.right, region.right);
3086         var b = Math.max(this.bottom, region.bottom);
3087         var l = Math.min(this.left, region.left);
3088
3089         return new Roo.lib.Region(t, r, b, l);
3090     },
3091
3092     adjust : function(t, l, b, r) {
3093         this.top += t;
3094         this.left += l;
3095         this.right += r;
3096         this.bottom += b;
3097         return this;
3098     }
3099 };
3100
3101 Roo.lib.Region.getRegion = function(el) {
3102     var p = Roo.lib.Dom.getXY(el);
3103
3104     var t = p[1];
3105     var r = p[0] + el.offsetWidth;
3106     var b = p[1] + el.offsetHeight;
3107     var l = p[0];
3108
3109     return new Roo.lib.Region(t, r, b, l);
3110 };
3111 /*
3112  * Portions of this file are based on pieces of Yahoo User Interface Library
3113  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3114  * YUI licensed under the BSD License:
3115  * http://developer.yahoo.net/yui/license.txt
3116  * <script type="text/javascript">
3117  *
3118  */
3119 //@@dep Roo.lib.Region
3120
3121
3122 Roo.lib.Point = function(x, y) {
3123     if (x instanceof Array) {
3124         y = x[1];
3125         x = x[0];
3126     }
3127     this.x = this.right = this.left = this[0] = x;
3128     this.y = this.top = this.bottom = this[1] = y;
3129 };
3130
3131 Roo.lib.Point.prototype = new Roo.lib.Region();
3132 /*
3133  * Portions of this file are based on pieces of Yahoo User Interface Library
3134  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3135  * YUI licensed under the BSD License:
3136  * http://developer.yahoo.net/yui/license.txt
3137  * <script type="text/javascript">
3138  *
3139  */
3140  
3141 (function() {   
3142
3143     Roo.lib.Anim = {
3144         scroll : function(el, args, duration, easing, cb, scope) {
3145             this.run(el, args, duration, easing, cb, scope, Roo.lib.Scroll);
3146         },
3147
3148         motion : function(el, args, duration, easing, cb, scope) {
3149             this.run(el, args, duration, easing, cb, scope, Roo.lib.Motion);
3150         },
3151
3152         color : function(el, args, duration, easing, cb, scope) {
3153             this.run(el, args, duration, easing, cb, scope, Roo.lib.ColorAnim);
3154         },
3155
3156         run : function(el, args, duration, easing, cb, scope, type) {
3157             type = type || Roo.lib.AnimBase;
3158             if (typeof easing == "string") {
3159                 easing = Roo.lib.Easing[easing];
3160             }
3161             var anim = new type(el, args, duration, easing);
3162             anim.animateX(function() {
3163                 Roo.callback(cb, scope);
3164             });
3165             return anim;
3166         }
3167     };
3168 })();/*
3169  * Portions of this file are based on pieces of Yahoo User Interface Library
3170  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3171  * YUI licensed under the BSD License:
3172  * http://developer.yahoo.net/yui/license.txt
3173  * <script type="text/javascript">
3174  *
3175  */
3176
3177 (function() {    
3178     var libFlyweight;
3179     
3180     function fly(el) {
3181         if (!libFlyweight) {
3182             libFlyweight = new Roo.Element.Flyweight();
3183         }
3184         libFlyweight.dom = el;
3185         return libFlyweight;
3186     }
3187
3188     // since this uses fly! - it cant be in DOM (which does not have fly yet..)
3189     
3190    
3191     
3192     Roo.lib.AnimBase = function(el, attributes, duration, method) {
3193         if (el) {
3194             this.init(el, attributes, duration, method);
3195         }
3196     };
3197
3198     Roo.lib.AnimBase.fly = fly;
3199     
3200     
3201     
3202     Roo.lib.AnimBase.prototype = {
3203
3204         toString: function() {
3205             var el = this.getEl();
3206             var id = el.id || el.tagName;
3207             return ("Anim " + id);
3208         },
3209
3210         patterns: {
3211             noNegatives:        /width|height|opacity|padding/i,
3212             offsetAttribute:  /^((width|height)|(top|left))$/,
3213             defaultUnit:        /width|height|top$|bottom$|left$|right$/i,
3214             offsetUnit:         /\d+(em|%|en|ex|pt|in|cm|mm|pc)$/i
3215         },
3216
3217
3218         doMethod: function(attr, start, end) {
3219             return this.method(this.currentFrame, start, end - start, this.totalFrames);
3220         },
3221
3222
3223         setAttribute: function(attr, val, unit) {
3224             if (this.patterns.noNegatives.test(attr)) {
3225                 val = (val > 0) ? val : 0;
3226             }
3227
3228             Roo.fly(this.getEl(), '_anim').setStyle(attr, val + unit);
3229         },
3230
3231
3232         getAttribute: function(attr) {
3233             var el = this.getEl();
3234             var val = fly(el).getStyle(attr);
3235
3236             if (val !== 'auto' && !this.patterns.offsetUnit.test(val)) {
3237                 return parseFloat(val);
3238             }
3239
3240             var a = this.patterns.offsetAttribute.exec(attr) || [];
3241             var pos = !!( a[3] );
3242             var box = !!( a[2] );
3243
3244
3245             if (box || (fly(el).getStyle('position') == 'absolute' && pos)) {
3246                 val = el['offset' + a[0].charAt(0).toUpperCase() + a[0].substr(1)];
3247             } else {
3248                 val = 0;
3249             }
3250
3251             return val;
3252         },
3253
3254
3255         getDefaultUnit: function(attr) {
3256             if (this.patterns.defaultUnit.test(attr)) {
3257                 return 'px';
3258             }
3259
3260             return '';
3261         },
3262
3263         animateX : function(callback, scope) {
3264             var f = function() {
3265                 this.onComplete.removeListener(f);
3266                 if (typeof callback == "function") {
3267                     callback.call(scope || this, this);
3268                 }
3269             };
3270             this.onComplete.addListener(f, this);
3271             this.animate();
3272         },
3273
3274
3275         setRuntimeAttribute: function(attr) {
3276             var start;
3277             var end;
3278             var attributes = this.attributes;
3279
3280             this.runtimeAttributes[attr] = {};
3281
3282             var isset = function(prop) {
3283                 return (typeof prop !== 'undefined');
3284             };
3285
3286             if (!isset(attributes[attr]['to']) && !isset(attributes[attr]['by'])) {
3287                 return false;
3288             }
3289
3290             start = ( isset(attributes[attr]['from']) ) ? attributes[attr]['from'] : this.getAttribute(attr);
3291
3292
3293             if (isset(attributes[attr]['to'])) {
3294                 end = attributes[attr]['to'];
3295             } else if (isset(attributes[attr]['by'])) {
3296                 if (start.constructor == Array) {
3297                     end = [];
3298                     for (var i = 0, len = start.length; i < len; ++i) {
3299                         end[i] = start[i] + attributes[attr]['by'][i];
3300                     }
3301                 } else {
3302                     end = start + attributes[attr]['by'];
3303                 }
3304             }
3305
3306             this.runtimeAttributes[attr].start = start;
3307             this.runtimeAttributes[attr].end = end;
3308
3309
3310             this.runtimeAttributes[attr].unit = ( isset(attributes[attr].unit) ) ? attributes[attr]['unit'] : this.getDefaultUnit(attr);
3311         },
3312
3313
3314         init: function(el, attributes, duration, method) {
3315
3316             var isAnimated = false;
3317
3318
3319             var startTime = null;
3320
3321
3322             var actualFrames = 0;
3323
3324
3325             el = Roo.getDom(el);
3326
3327
3328             this.attributes = attributes || {};
3329
3330
3331             this.duration = duration || 1;
3332
3333
3334             this.method = method || Roo.lib.Easing.easeNone;
3335
3336
3337             this.useSeconds = true;
3338
3339
3340             this.currentFrame = 0;
3341
3342
3343             this.totalFrames = Roo.lib.AnimMgr.fps;
3344
3345
3346             this.getEl = function() {
3347                 return el;
3348             };
3349
3350
3351             this.isAnimated = function() {
3352                 return isAnimated;
3353             };
3354
3355
3356             this.getStartTime = function() {
3357                 return startTime;
3358             };
3359
3360             this.runtimeAttributes = {};
3361
3362
3363             this.animate = function() {
3364                 if (this.isAnimated()) {
3365                     return false;
3366                 }
3367
3368                 this.currentFrame = 0;
3369
3370                 this.totalFrames = ( this.useSeconds ) ? Math.ceil(Roo.lib.AnimMgr.fps * this.duration) : this.duration;
3371
3372                 Roo.lib.AnimMgr.registerElement(this);
3373             };
3374
3375
3376             this.stop = function(finish) {
3377                 if (finish) {
3378                     this.currentFrame = this.totalFrames;
3379                     this._onTween.fire();
3380                 }
3381                 Roo.lib.AnimMgr.stop(this);
3382             };
3383
3384             var onStart = function() {
3385                 this.onStart.fire();
3386
3387                 this.runtimeAttributes = {};
3388                 for (var attr in this.attributes) {
3389                     this.setRuntimeAttribute(attr);
3390                 }
3391
3392                 isAnimated = true;
3393                 actualFrames = 0;
3394                 startTime = new Date();
3395             };
3396
3397
3398             var onTween = function() {
3399                 var data = {
3400                     duration: new Date() - this.getStartTime(),
3401                     currentFrame: this.currentFrame
3402                 };
3403
3404                 data.toString = function() {
3405                     return (
3406                             'duration: ' + data.duration +
3407                             ', currentFrame: ' + data.currentFrame
3408                             );
3409                 };
3410
3411                 this.onTween.fire(data);
3412
3413                 var runtimeAttributes = this.runtimeAttributes;
3414
3415                 for (var attr in runtimeAttributes) {
3416                     this.setAttribute(attr, this.doMethod(attr, runtimeAttributes[attr].start, runtimeAttributes[attr].end), runtimeAttributes[attr].unit);
3417                 }
3418
3419                 actualFrames += 1;
3420             };
3421
3422             var onComplete = function() {
3423                 var actual_duration = (new Date() - startTime) / 1000 ;
3424
3425                 var data = {
3426                     duration: actual_duration,
3427                     frames: actualFrames,
3428                     fps: actualFrames / actual_duration
3429                 };
3430
3431                 data.toString = function() {
3432                     return (
3433                             'duration: ' + data.duration +
3434                             ', frames: ' + data.frames +
3435                             ', fps: ' + data.fps
3436                             );
3437                 };
3438
3439                 isAnimated = false;
3440                 actualFrames = 0;
3441                 this.onComplete.fire(data);
3442             };
3443
3444
3445             this._onStart = new Roo.util.Event(this);
3446             this.onStart = new Roo.util.Event(this);
3447             this.onTween = new Roo.util.Event(this);
3448             this._onTween = new Roo.util.Event(this);
3449             this.onComplete = new Roo.util.Event(this);
3450             this._onComplete = new Roo.util.Event(this);
3451             this._onStart.addListener(onStart);
3452             this._onTween.addListener(onTween);
3453             this._onComplete.addListener(onComplete);
3454         }
3455     };
3456 })();
3457 /*
3458  * Portions of this file are based on pieces of Yahoo User Interface Library
3459  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3460  * YUI licensed under the BSD License:
3461  * http://developer.yahoo.net/yui/license.txt
3462  * <script type="text/javascript">
3463  *
3464  */
3465
3466 Roo.lib.AnimMgr = new function() {
3467
3468     var thread = null;
3469
3470
3471     var queue = [];
3472
3473
3474     var tweenCount = 0;
3475
3476
3477     this.fps = 1000;
3478
3479
3480     this.delay = 1;
3481
3482
3483     this.registerElement = function(tween) {
3484         queue[queue.length] = tween;
3485         tweenCount += 1;
3486         tween._onStart.fire();
3487         this.start();
3488     };
3489
3490
3491     this.unRegister = function(tween, index) {
3492         tween._onComplete.fire();
3493         index = index || getIndex(tween);
3494         if (index != -1) {
3495             queue.splice(index, 1);
3496         }
3497
3498         tweenCount -= 1;
3499         if (tweenCount <= 0) {
3500             this.stop();
3501         }
3502     };
3503
3504
3505     this.start = function() {
3506         if (thread === null) {
3507             thread = setInterval(this.run, this.delay);
3508         }
3509     };
3510
3511
3512     this.stop = function(tween) {
3513         if (!tween) {
3514             clearInterval(thread);
3515
3516             for (var i = 0, len = queue.length; i < len; ++i) {
3517                 if (queue[0].isAnimated()) {
3518                     this.unRegister(queue[0], 0);
3519                 }
3520             }
3521
3522             queue = [];
3523             thread = null;
3524             tweenCount = 0;
3525         }
3526         else {
3527             this.unRegister(tween);
3528         }
3529     };
3530
3531
3532     this.run = function() {
3533         for (var i = 0, len = queue.length; i < len; ++i) {
3534             var tween = queue[i];
3535             if (!tween || !tween.isAnimated()) {
3536                 continue;
3537             }
3538
3539             if (tween.currentFrame < tween.totalFrames || tween.totalFrames === null)
3540             {
3541                 tween.currentFrame += 1;
3542
3543                 if (tween.useSeconds) {
3544                     correctFrame(tween);
3545                 }
3546                 tween._onTween.fire();
3547             }
3548             else {
3549                 Roo.lib.AnimMgr.stop(tween, i);
3550             }
3551         }
3552     };
3553
3554     var getIndex = function(anim) {
3555         for (var i = 0, len = queue.length; i < len; ++i) {
3556             if (queue[i] == anim) {
3557                 return i;
3558             }
3559         }
3560         return -1;
3561     };
3562
3563
3564     var correctFrame = function(tween) {
3565         var frames = tween.totalFrames;
3566         var frame = tween.currentFrame;
3567         var expected = (tween.currentFrame * tween.duration * 1000 / tween.totalFrames);
3568         var elapsed = (new Date() - tween.getStartTime());
3569         var tweak = 0;
3570
3571         if (elapsed < tween.duration * 1000) {
3572             tweak = Math.round((elapsed / expected - 1) * tween.currentFrame);
3573         } else {
3574             tweak = frames - (frame + 1);
3575         }
3576         if (tweak > 0 && isFinite(tweak)) {
3577             if (tween.currentFrame + tweak >= frames) {
3578                 tweak = frames - (frame + 1);
3579             }
3580
3581             tween.currentFrame += tweak;
3582         }
3583     };
3584 };
3585
3586     /*
3587  * Portions of this file are based on pieces of Yahoo User Interface Library
3588  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3589  * YUI licensed under the BSD License:
3590  * http://developer.yahoo.net/yui/license.txt
3591  * <script type="text/javascript">
3592  *
3593  */
3594 Roo.lib.Bezier = new function() {
3595
3596         this.getPosition = function(points, t) {
3597             var n = points.length;
3598             var tmp = [];
3599
3600             for (var i = 0; i < n; ++i) {
3601                 tmp[i] = [points[i][0], points[i][1]];
3602             }
3603
3604             for (var j = 1; j < n; ++j) {
3605                 for (i = 0; i < n - j; ++i) {
3606                     tmp[i][0] = (1 - t) * tmp[i][0] + t * tmp[parseInt(i + 1, 10)][0];
3607                     tmp[i][1] = (1 - t) * tmp[i][1] + t * tmp[parseInt(i + 1, 10)][1];
3608                 }
3609             }
3610
3611             return [ tmp[0][0], tmp[0][1] ];
3612
3613         };
3614     };/*
3615  * Portions of this file are based on pieces of Yahoo User Interface Library
3616  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3617  * YUI licensed under the BSD License:
3618  * http://developer.yahoo.net/yui/license.txt
3619  * <script type="text/javascript">
3620  *
3621  */
3622 (function() {
3623
3624     Roo.lib.ColorAnim = function(el, attributes, duration, method) {
3625         Roo.lib.ColorAnim.superclass.constructor.call(this, el, attributes, duration, method);
3626     };
3627
3628     Roo.extend(Roo.lib.ColorAnim, Roo.lib.AnimBase);
3629
3630     var fly = Roo.lib.AnimBase.fly;
3631     var Y = Roo.lib;
3632     var superclass = Y.ColorAnim.superclass;
3633     var proto = Y.ColorAnim.prototype;
3634
3635     proto.toString = function() {
3636         var el = this.getEl();
3637         var id = el.id || el.tagName;
3638         return ("ColorAnim " + id);
3639     };
3640
3641     proto.patterns.color = /color$/i;
3642     proto.patterns.rgb = /^rgb\(([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\)$/i;
3643     proto.patterns.hex = /^#?([0-9A-F]{2})([0-9A-F]{2})([0-9A-F]{2})$/i;
3644     proto.patterns.hex3 = /^#?([0-9A-F]{1})([0-9A-F]{1})([0-9A-F]{1})$/i;
3645     proto.patterns.transparent = /^transparent|rgba\(0, 0, 0, 0\)$/;
3646
3647
3648     proto.parseColor = function(s) {
3649         if (s.length == 3) {
3650             return s;
3651         }
3652
3653         var c = this.patterns.hex.exec(s);
3654         if (c && c.length == 4) {
3655             return [ parseInt(c[1], 16), parseInt(c[2], 16), parseInt(c[3], 16) ];
3656         }
3657
3658         c = this.patterns.rgb.exec(s);
3659         if (c && c.length == 4) {
3660             return [ parseInt(c[1], 10), parseInt(c[2], 10), parseInt(c[3], 10) ];
3661         }
3662
3663         c = this.patterns.hex3.exec(s);
3664         if (c && c.length == 4) {
3665             return [ parseInt(c[1] + c[1], 16), parseInt(c[2] + c[2], 16), parseInt(c[3] + c[3], 16) ];
3666         }
3667
3668         return null;
3669     };
3670     // since this uses fly! - it cant be in ColorAnim (which does not have fly yet..)
3671     proto.getAttribute = function(attr) {
3672         var el = this.getEl();
3673         if (this.patterns.color.test(attr)) {
3674             var val = fly(el).getStyle(attr);
3675
3676             if (this.patterns.transparent.test(val)) {
3677                 var parent = el.parentNode;
3678                 val = fly(parent).getStyle(attr);
3679
3680                 while (parent && this.patterns.transparent.test(val)) {
3681                     parent = parent.parentNode;
3682                     val = fly(parent).getStyle(attr);
3683                     if (parent.tagName.toUpperCase() == 'HTML') {
3684                         val = '#fff';
3685                     }
3686                 }
3687             }
3688         } else {
3689             val = superclass.getAttribute.call(this, attr);
3690         }
3691
3692         return val;
3693     };
3694     proto.getAttribute = function(attr) {
3695         var el = this.getEl();
3696         if (this.patterns.color.test(attr)) {
3697             var val = fly(el).getStyle(attr);
3698
3699             if (this.patterns.transparent.test(val)) {
3700                 var parent = el.parentNode;
3701                 val = fly(parent).getStyle(attr);
3702
3703                 while (parent && this.patterns.transparent.test(val)) {
3704                     parent = parent.parentNode;
3705                     val = fly(parent).getStyle(attr);
3706                     if (parent.tagName.toUpperCase() == 'HTML') {
3707                         val = '#fff';
3708                     }
3709                 }
3710             }
3711         } else {
3712             val = superclass.getAttribute.call(this, attr);
3713         }
3714
3715         return val;
3716     };
3717
3718     proto.doMethod = function(attr, start, end) {
3719         var val;
3720
3721         if (this.patterns.color.test(attr)) {
3722             val = [];
3723             for (var i = 0, len = start.length; i < len; ++i) {
3724                 val[i] = superclass.doMethod.call(this, attr, start[i], end[i]);
3725             }
3726
3727             val = 'rgb(' + Math.floor(val[0]) + ',' + Math.floor(val[1]) + ',' + Math.floor(val[2]) + ')';
3728         }
3729         else {
3730             val = superclass.doMethod.call(this, attr, start, end);
3731         }
3732
3733         return val;
3734     };
3735
3736     proto.setRuntimeAttribute = function(attr) {
3737         superclass.setRuntimeAttribute.call(this, attr);
3738
3739         if (this.patterns.color.test(attr)) {
3740             var attributes = this.attributes;
3741             var start = this.parseColor(this.runtimeAttributes[attr].start);
3742             var end = this.parseColor(this.runtimeAttributes[attr].end);
3743
3744             if (typeof attributes[attr]['to'] === 'undefined' && typeof attributes[attr]['by'] !== 'undefined') {
3745                 end = this.parseColor(attributes[attr].by);
3746
3747                 for (var i = 0, len = start.length; i < len; ++i) {
3748                     end[i] = start[i] + end[i];
3749                 }
3750             }
3751
3752             this.runtimeAttributes[attr].start = start;
3753             this.runtimeAttributes[attr].end = end;
3754         }
3755     };
3756 })();
3757
3758 /*
3759  * Portions of this file are based on pieces of Yahoo User Interface Library
3760  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3761  * YUI licensed under the BSD License:
3762  * http://developer.yahoo.net/yui/license.txt
3763  * <script type="text/javascript">
3764  *
3765  */
3766 Roo.lib.Easing = {
3767
3768
3769     easeNone: function (t, b, c, d) {
3770         return c * t / d + b;
3771     },
3772
3773
3774     easeIn: function (t, b, c, d) {
3775         return c * (t /= d) * t + b;
3776     },
3777
3778
3779     easeOut: function (t, b, c, d) {
3780         return -c * (t /= d) * (t - 2) + b;
3781     },
3782
3783
3784     easeBoth: function (t, b, c, d) {
3785         if ((t /= d / 2) < 1) {
3786             return c / 2 * t * t + b;
3787         }
3788
3789         return -c / 2 * ((--t) * (t - 2) - 1) + b;
3790     },
3791
3792
3793     easeInStrong: function (t, b, c, d) {
3794         return c * (t /= d) * t * t * t + b;
3795     },
3796
3797
3798     easeOutStrong: function (t, b, c, d) {
3799         return -c * ((t = t / d - 1) * t * t * t - 1) + b;
3800     },
3801
3802
3803     easeBothStrong: function (t, b, c, d) {
3804         if ((t /= d / 2) < 1) {
3805             return c / 2 * t * t * t * t + b;
3806         }
3807
3808         return -c / 2 * ((t -= 2) * t * t * t - 2) + b;
3809     },
3810
3811
3812
3813     elasticIn: function (t, b, c, d, a, p) {
3814         if (t == 0) {
3815             return b;
3816         }
3817         if ((t /= d) == 1) {
3818             return b + c;
3819         }
3820         if (!p) {
3821             p = d * .3;
3822         }
3823
3824         if (!a || a < Math.abs(c)) {
3825             a = c;
3826             var s = p / 4;
3827         }
3828         else {
3829             var s = p / (2 * Math.PI) * Math.asin(c / a);
3830         }
3831
3832         return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3833     },
3834
3835
3836     elasticOut: function (t, b, c, d, a, p) {
3837         if (t == 0) {
3838             return b;
3839         }
3840         if ((t /= d) == 1) {
3841             return b + c;
3842         }
3843         if (!p) {
3844             p = d * .3;
3845         }
3846
3847         if (!a || a < Math.abs(c)) {
3848             a = c;
3849             var s = p / 4;
3850         }
3851         else {
3852             var s = p / (2 * Math.PI) * Math.asin(c / a);
3853         }
3854
3855         return a * Math.pow(2, -10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b;
3856     },
3857
3858
3859     elasticBoth: function (t, b, c, d, a, p) {
3860         if (t == 0) {
3861             return b;
3862         }
3863
3864         if ((t /= d / 2) == 2) {
3865             return b + c;
3866         }
3867
3868         if (!p) {
3869             p = d * (.3 * 1.5);
3870         }
3871
3872         if (!a || a < Math.abs(c)) {
3873             a = c;
3874             var s = p / 4;
3875         }
3876         else {
3877             var s = p / (2 * Math.PI) * Math.asin(c / a);
3878         }
3879
3880         if (t < 1) {
3881             return -.5 * (a * Math.pow(2, 10 * (t -= 1)) *
3882                           Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3883         }
3884         return a * Math.pow(2, -10 * (t -= 1)) *
3885                Math.sin((t * d - s) * (2 * Math.PI) / p) * .5 + c + b;
3886     },
3887
3888
3889
3890     backIn: function (t, b, c, d, s) {
3891         if (typeof s == 'undefined') {
3892             s = 1.70158;
3893         }
3894         return c * (t /= d) * t * ((s + 1) * t - s) + b;
3895     },
3896
3897
3898     backOut: function (t, b, c, d, s) {
3899         if (typeof s == 'undefined') {
3900             s = 1.70158;
3901         }
3902         return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
3903     },
3904
3905
3906     backBoth: function (t, b, c, d, s) {
3907         if (typeof s == 'undefined') {
3908             s = 1.70158;
3909         }
3910
3911         if ((t /= d / 2 ) < 1) {
3912             return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
3913         }
3914         return c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
3915     },
3916
3917
3918     bounceIn: function (t, b, c, d) {
3919         return c - Roo.lib.Easing.bounceOut(d - t, 0, c, d) + b;
3920     },
3921
3922
3923     bounceOut: function (t, b, c, d) {
3924         if ((t /= d) < (1 / 2.75)) {
3925             return c * (7.5625 * t * t) + b;
3926         } else if (t < (2 / 2.75)) {
3927             return c * (7.5625 * (t -= (1.5 / 2.75)) * t + .75) + b;
3928         } else if (t < (2.5 / 2.75)) {
3929             return c * (7.5625 * (t -= (2.25 / 2.75)) * t + .9375) + b;
3930         }
3931         return c * (7.5625 * (t -= (2.625 / 2.75)) * t + .984375) + b;
3932     },
3933
3934
3935     bounceBoth: function (t, b, c, d) {
3936         if (t < d / 2) {
3937             return Roo.lib.Easing.bounceIn(t * 2, 0, c, d) * .5 + b;
3938         }
3939         return Roo.lib.Easing.bounceOut(t * 2 - d, 0, c, d) * .5 + c * .5 + b;
3940     }
3941 };/*
3942  * Portions of this file are based on pieces of Yahoo User Interface Library
3943  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3944  * YUI licensed under the BSD License:
3945  * http://developer.yahoo.net/yui/license.txt
3946  * <script type="text/javascript">
3947  *
3948  */
3949     (function() {
3950         Roo.lib.Motion = function(el, attributes, duration, method) {
3951             if (el) {
3952                 Roo.lib.Motion.superclass.constructor.call(this, el, attributes, duration, method);
3953             }
3954         };
3955
3956         Roo.extend(Roo.lib.Motion, Roo.lib.ColorAnim);
3957
3958
3959         var Y = Roo.lib;
3960         var superclass = Y.Motion.superclass;
3961         var proto = Y.Motion.prototype;
3962
3963         proto.toString = function() {
3964             var el = this.getEl();
3965             var id = el.id || el.tagName;
3966             return ("Motion " + id);
3967         };
3968
3969         proto.patterns.points = /^points$/i;
3970
3971         proto.setAttribute = function(attr, val, unit) {
3972             if (this.patterns.points.test(attr)) {
3973                 unit = unit || 'px';
3974                 superclass.setAttribute.call(this, 'left', val[0], unit);
3975                 superclass.setAttribute.call(this, 'top', val[1], unit);
3976             } else {
3977                 superclass.setAttribute.call(this, attr, val, unit);
3978             }
3979         };
3980
3981         proto.getAttribute = function(attr) {
3982             if (this.patterns.points.test(attr)) {
3983                 var val = [
3984                         superclass.getAttribute.call(this, 'left'),
3985                         superclass.getAttribute.call(this, 'top')
3986                         ];
3987             } else {
3988                 val = superclass.getAttribute.call(this, attr);
3989             }
3990
3991             return val;
3992         };
3993
3994         proto.doMethod = function(attr, start, end) {
3995             var val = null;
3996
3997             if (this.patterns.points.test(attr)) {
3998                 var t = this.method(this.currentFrame, 0, 100, this.totalFrames) / 100;
3999                 val = Y.Bezier.getPosition(this.runtimeAttributes[attr], t);
4000             } else {
4001                 val = superclass.doMethod.call(this, attr, start, end);
4002             }
4003             return val;
4004         };
4005
4006         proto.setRuntimeAttribute = function(attr) {
4007             if (this.patterns.points.test(attr)) {
4008                 var el = this.getEl();
4009                 var attributes = this.attributes;
4010                 var start;
4011                 var control = attributes['points']['control'] || [];
4012                 var end;
4013                 var i, len;
4014
4015                 if (control.length > 0 && !(control[0] instanceof Array)) {
4016                     control = [control];
4017                 } else {
4018                     var tmp = [];
4019                     for (i = 0,len = control.length; i < len; ++i) {
4020                         tmp[i] = control[i];
4021                     }
4022                     control = tmp;
4023                 }
4024
4025                 Roo.fly(el).position();
4026
4027                 if (isset(attributes['points']['from'])) {
4028                     Roo.lib.Dom.setXY(el, attributes['points']['from']);
4029                 }
4030                 else {
4031                     Roo.lib.Dom.setXY(el, Roo.lib.Dom.getXY(el));
4032                 }
4033
4034                 start = this.getAttribute('points');
4035
4036
4037                 if (isset(attributes['points']['to'])) {
4038                     end = translateValues.call(this, attributes['points']['to'], start);
4039
4040                     var pageXY = Roo.lib.Dom.getXY(this.getEl());
4041                     for (i = 0,len = control.length; i < len; ++i) {
4042                         control[i] = translateValues.call(this, control[i], start);
4043                     }
4044
4045
4046                 } else if (isset(attributes['points']['by'])) {
4047                     end = [ start[0] + attributes['points']['by'][0], start[1] + attributes['points']['by'][1] ];
4048
4049                     for (i = 0,len = control.length; i < len; ++i) {
4050                         control[i] = [ start[0] + control[i][0], start[1] + control[i][1] ];
4051                     }
4052                 }
4053
4054                 this.runtimeAttributes[attr] = [start];
4055
4056                 if (control.length > 0) {
4057                     this.runtimeAttributes[attr] = this.runtimeAttributes[attr].concat(control);
4058                 }
4059
4060                 this.runtimeAttributes[attr][this.runtimeAttributes[attr].length] = end;
4061             }
4062             else {
4063                 superclass.setRuntimeAttribute.call(this, attr);
4064             }
4065         };
4066
4067         var translateValues = function(val, start) {
4068             var pageXY = Roo.lib.Dom.getXY(this.getEl());
4069             val = [ val[0] - pageXY[0] + start[0], val[1] - pageXY[1] + start[1] ];
4070
4071             return val;
4072         };
4073
4074         var isset = function(prop) {
4075             return (typeof prop !== 'undefined');
4076         };
4077     })();
4078 /*
4079  * Portions of this file are based on pieces of Yahoo User Interface Library
4080  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
4081  * YUI licensed under the BSD License:
4082  * http://developer.yahoo.net/yui/license.txt
4083  * <script type="text/javascript">
4084  *
4085  */
4086     (function() {
4087         Roo.lib.Scroll = function(el, attributes, duration, method) {
4088             if (el) {
4089                 Roo.lib.Scroll.superclass.constructor.call(this, el, attributes, duration, method);
4090             }
4091         };
4092
4093         Roo.extend(Roo.lib.Scroll, Roo.lib.ColorAnim);
4094
4095
4096         var Y = Roo.lib;
4097         var superclass = Y.Scroll.superclass;
4098         var proto = Y.Scroll.prototype;
4099
4100         proto.toString = function() {
4101             var el = this.getEl();
4102             var id = el.id || el.tagName;
4103             return ("Scroll " + id);
4104         };
4105
4106         proto.doMethod = function(attr, start, end) {
4107             var val = null;
4108
4109             if (attr == 'scroll') {
4110                 val = [
4111                         this.method(this.currentFrame, start[0], end[0] - start[0], this.totalFrames),
4112                         this.method(this.currentFrame, start[1], end[1] - start[1], this.totalFrames)
4113                         ];
4114
4115             } else {
4116                 val = superclass.doMethod.call(this, attr, start, end);
4117             }
4118             return val;
4119         };
4120
4121         proto.getAttribute = function(attr) {
4122             var val = null;
4123             var el = this.getEl();
4124
4125             if (attr == 'scroll') {
4126                 val = [ el.scrollLeft, el.scrollTop ];
4127             } else {
4128                 val = superclass.getAttribute.call(this, attr);
4129             }
4130
4131             return val;
4132         };
4133
4134         proto.setAttribute = function(attr, val, unit) {
4135             var el = this.getEl();
4136
4137             if (attr == 'scroll') {
4138                 el.scrollLeft = val[0];
4139                 el.scrollTop = val[1];
4140             } else {
4141                 superclass.setAttribute.call(this, attr, val, unit);
4142             }
4143         };
4144     })();
4145 /*
4146  * Based on:
4147  * Ext JS Library 1.1.1
4148  * Copyright(c) 2006-2007, Ext JS, LLC.
4149  *
4150  * Originally Released Under LGPL - original licence link has changed is not relivant.
4151  *
4152  * Fork - LGPL
4153  * <script type="text/javascript">
4154  */
4155
4156
4157 // nasty IE9 hack - what a pile of crap that is..
4158
4159  if (typeof Range != "undefined" && typeof Range.prototype.createContextualFragment == "undefined") {
4160     Range.prototype.createContextualFragment = function (html) {
4161         var doc = window.document;
4162         var container = doc.createElement("div");
4163         container.innerHTML = html;
4164         var frag = doc.createDocumentFragment(), n;
4165         while ((n = container.firstChild)) {
4166             frag.appendChild(n);
4167         }
4168         return frag;
4169     };
4170 }
4171
4172 /**
4173  * @class Roo.DomHelper
4174  * Utility class for working with DOM and/or Templates. It transparently supports using HTML fragments or DOM.
4175  * For more information see <a href="http://web.archive.org/web/20071221063734/http://www.jackslocum.com/blog/2006/10/06/domhelper-create-elements-using-dom-html-fragments-or-templates/">this blog post with examples</a>.
4176  * @singleton
4177  */
4178 Roo.DomHelper = function(){
4179     var tempTableEl = null;
4180     var emptyTags = /^(?:br|frame|hr|img|input|link|meta|range|spacer|wbr|area|param|col)$/i;
4181     var tableRe = /^table|tbody|tr|td$/i;
4182     var xmlns = {};
4183     // build as innerHTML where available
4184     /** @ignore */
4185     var createHtml = function(o){
4186         if(typeof o == 'string'){
4187             return o;
4188         }
4189         var b = "";
4190         if(!o.tag){
4191             o.tag = "div";
4192         }
4193         b += "<" + o.tag;
4194         for(var attr in o){
4195             if(attr == "tag" || attr == "children" || attr == "cn" || attr == "html" || typeof o[attr] == "function") { continue; }
4196             if(attr == "style"){
4197                 var s = o["style"];
4198                 if(typeof s == "function"){
4199                     s = s.call();
4200                 }
4201                 if(typeof s == "string"){
4202                     b += ' style="' + s + '"';
4203                 }else if(typeof s == "object"){
4204                     b += ' style="';
4205                     for(var key in s){
4206                         if(typeof s[key] != "function"){
4207                             b += key + ":" + s[key] + ";";
4208                         }
4209                     }
4210                     b += '"';
4211                 }
4212             }else{
4213                 if(attr == "cls"){
4214                     b += ' class="' + o["cls"] + '"';
4215                 }else if(attr == "htmlFor"){
4216                     b += ' for="' + o["htmlFor"] + '"';
4217                 }else{
4218                     b += " " + attr + '="' + o[attr] + '"';
4219                 }
4220             }
4221         }
4222         if(emptyTags.test(o.tag)){
4223             b += "/>";
4224         }else{
4225             b += ">";
4226             var cn = o.children || o.cn;
4227             if(cn){
4228                 //http://bugs.kde.org/show_bug.cgi?id=71506
4229                 if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4230                     for(var i = 0, len = cn.length; i < len; i++) {
4231                         b += createHtml(cn[i], b);
4232                     }
4233                 }else{
4234                     b += createHtml(cn, b);
4235                 }
4236             }
4237             if(o.html){
4238                 b += o.html;
4239             }
4240             b += "</" + o.tag + ">";
4241         }
4242         return b;
4243     };
4244
4245     // build as dom
4246     /** @ignore */
4247     var createDom = function(o, parentNode){
4248          
4249         // defininition craeted..
4250         var ns = false;
4251         if (o.ns && o.ns != 'html') {
4252                
4253             if (o.xmlns && typeof(xmlns[o.ns]) == 'undefined') {
4254                 xmlns[o.ns] = o.xmlns;
4255                 ns = o.xmlns;
4256             }
4257             if (typeof(xmlns[o.ns]) == 'undefined') {
4258                 console.log("Trying to create namespace element " + o.ns + ", however no xmlns was sent to builder previously");
4259             }
4260             ns = xmlns[o.ns];
4261         }
4262         
4263         
4264         if (typeof(o) == 'string') {
4265             return parentNode.appendChild(document.createTextNode(o));
4266         }
4267         o.tag = o.tag || div;
4268         if (o.ns && Roo.isIE) {
4269             ns = false;
4270             o.tag = o.ns + ':' + o.tag;
4271             
4272         }
4273         var el = ns ? document.createElementNS( ns, o.tag||'div') :  document.createElement(o.tag||'div');
4274         var useSet = el.setAttribute ? true : false; // In IE some elements don't have setAttribute
4275         for(var attr in o){
4276             
4277             if(attr == "tag" || attr == "ns" ||attr == "xmlns" ||attr == "children" || attr == "cn" || attr == "html" || 
4278                     attr == "style" || typeof o[attr] == "function") { continue; }
4279                     
4280             if(attr=="cls" && Roo.isIE){
4281                 el.className = o["cls"];
4282             }else{
4283                 if(useSet) { el.setAttribute(attr=="cls" ? 'class' : attr, o[attr]);}
4284                 else { 
4285                     el[attr] = o[attr];
4286                 }
4287             }
4288         }
4289         Roo.DomHelper.applyStyles(el, o.style);
4290         var cn = o.children || o.cn;
4291         if(cn){
4292             //http://bugs.kde.org/show_bug.cgi?id=71506
4293              if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4294                 for(var i = 0, len = cn.length; i < len; i++) {
4295                     createDom(cn[i], el);
4296                 }
4297             }else{
4298                 createDom(cn, el);
4299             }
4300         }
4301         if(o.html){
4302             el.innerHTML = o.html;
4303         }
4304         if(parentNode){
4305            parentNode.appendChild(el);
4306         }
4307         return el;
4308     };
4309
4310     var ieTable = function(depth, s, h, e){
4311         tempTableEl.innerHTML = [s, h, e].join('');
4312         var i = -1, el = tempTableEl;
4313         while(++i < depth){
4314             el = el.firstChild;
4315         }
4316         return el;
4317     };
4318
4319     // kill repeat to save bytes
4320     var ts = '<table>',
4321         te = '</table>',
4322         tbs = ts+'<tbody>',
4323         tbe = '</tbody>'+te,
4324         trs = tbs + '<tr>',
4325         tre = '</tr>'+tbe;
4326
4327     /**
4328      * @ignore
4329      * Nasty code for IE's broken table implementation
4330      */
4331     var insertIntoTable = function(tag, where, el, html){
4332         if(!tempTableEl){
4333             tempTableEl = document.createElement('div');
4334         }
4335         var node;
4336         var before = null;
4337         if(tag == 'td'){
4338             if(where == 'afterbegin' || where == 'beforeend'){ // INTO a TD
4339                 return;
4340             }
4341             if(where == 'beforebegin'){
4342                 before = el;
4343                 el = el.parentNode;
4344             } else{
4345                 before = el.nextSibling;
4346                 el = el.parentNode;
4347             }
4348             node = ieTable(4, trs, html, tre);
4349         }
4350         else if(tag == 'tr'){
4351             if(where == 'beforebegin'){
4352                 before = el;
4353                 el = el.parentNode;
4354                 node = ieTable(3, tbs, html, tbe);
4355             } else if(where == 'afterend'){
4356                 before = el.nextSibling;
4357                 el = el.parentNode;
4358                 node = ieTable(3, tbs, html, tbe);
4359             } else{ // INTO a TR
4360                 if(where == 'afterbegin'){
4361                     before = el.firstChild;
4362                 }
4363                 node = ieTable(4, trs, html, tre);
4364             }
4365         } else if(tag == 'tbody'){
4366             if(where == 'beforebegin'){
4367                 before = el;
4368                 el = el.parentNode;
4369                 node = ieTable(2, ts, html, te);
4370             } else if(where == 'afterend'){
4371                 before = el.nextSibling;
4372                 el = el.parentNode;
4373                 node = ieTable(2, ts, html, te);
4374             } else{
4375                 if(where == 'afterbegin'){
4376                     before = el.firstChild;
4377                 }
4378                 node = ieTable(3, tbs, html, tbe);
4379             }
4380         } else{ // TABLE
4381             if(where == 'beforebegin' || where == 'afterend'){ // OUTSIDE the table
4382                 return;
4383             }
4384             if(where == 'afterbegin'){
4385                 before = el.firstChild;
4386             }
4387             node = ieTable(2, ts, html, te);
4388         }
4389         el.insertBefore(node, before);
4390         return node;
4391     };
4392
4393     return {
4394     /** True to force the use of DOM instead of html fragments @type Boolean */
4395     useDom : false,
4396
4397     /**
4398      * Returns the markup for the passed Element(s) config
4399      * @param {Object} o The Dom object spec (and children)
4400      * @return {String}
4401      */
4402     markup : function(o){
4403         return createHtml(o);
4404     },
4405
4406     /**
4407      * Applies a style specification to an element
4408      * @param {String/HTMLElement} el The element to apply styles to
4409      * @param {String/Object/Function} styles A style specification string eg "width:100px", or object in the form {width:"100px"}, or
4410      * a function which returns such a specification.
4411      */
4412     applyStyles : function(el, styles){
4413         if(styles){
4414            el = Roo.fly(el);
4415            if(typeof styles == "string"){
4416                var re = /\s?([a-z\-]*)\:\s?([^;]*);?/gi;
4417                var matches;
4418                while ((matches = re.exec(styles)) != null){
4419                    el.setStyle(matches[1], matches[2]);
4420                }
4421            }else if (typeof styles == "object"){
4422                for (var style in styles){
4423                   el.setStyle(style, styles[style]);
4424                }
4425            }else if (typeof styles == "function"){
4426                 Roo.DomHelper.applyStyles(el, styles.call());
4427            }
4428         }
4429     },
4430
4431     /**
4432      * Inserts an HTML fragment into the Dom
4433      * @param {String} where Where to insert the html in relation to el - beforeBegin, afterBegin, beforeEnd, afterEnd.
4434      * @param {HTMLElement} el The context element
4435      * @param {String} html The HTML fragmenet
4436      * @return {HTMLElement} The new node
4437      */
4438     insertHtml : function(where, el, html){
4439         where = where.toLowerCase();
4440         if(el.insertAdjacentHTML){
4441             if(tableRe.test(el.tagName)){
4442                 var rs;
4443                 if(rs = insertIntoTable(el.tagName.toLowerCase(), where, el, html)){
4444                     return rs;
4445                 }
4446             }
4447             switch(where){
4448                 case "beforebegin":
4449                     el.insertAdjacentHTML('BeforeBegin', html);
4450                     return el.previousSibling;
4451                 case "afterbegin":
4452                     el.insertAdjacentHTML('AfterBegin', html);
4453                     return el.firstChild;
4454                 case "beforeend":
4455                     el.insertAdjacentHTML('BeforeEnd', html);
4456                     return el.lastChild;
4457                 case "afterend":
4458                     el.insertAdjacentHTML('AfterEnd', html);
4459                     return el.nextSibling;
4460             }
4461             throw 'Illegal insertion point -> "' + where + '"';
4462         }
4463         var range = el.ownerDocument.createRange();
4464         var frag;
4465         switch(where){
4466              case "beforebegin":
4467                 range.setStartBefore(el);
4468                 frag = range.createContextualFragment(html);
4469                 el.parentNode.insertBefore(frag, el);
4470                 return el.previousSibling;
4471              case "afterbegin":
4472                 if(el.firstChild){
4473                     range.setStartBefore(el.firstChild);
4474                     frag = range.createContextualFragment(html);
4475                     el.insertBefore(frag, el.firstChild);
4476                     return el.firstChild;
4477                 }else{
4478                     el.innerHTML = html;
4479                     return el.firstChild;
4480                 }
4481             case "beforeend":
4482                 if(el.lastChild){
4483                     range.setStartAfter(el.lastChild);
4484                     frag = range.createContextualFragment(html);
4485                     el.appendChild(frag);
4486                     return el.lastChild;
4487                 }else{
4488                     el.innerHTML = html;
4489                     return el.lastChild;
4490                 }
4491             case "afterend":
4492                 range.setStartAfter(el);
4493                 frag = range.createContextualFragment(html);
4494                 el.parentNode.insertBefore(frag, el.nextSibling);
4495                 return el.nextSibling;
4496             }
4497             throw 'Illegal insertion point -> "' + where + '"';
4498     },
4499
4500     /**
4501      * Creates new Dom element(s) and inserts them before el
4502      * @param {String/HTMLElement/Element} el The context element
4503      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4504      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4505      * @return {HTMLElement/Roo.Element} The new node
4506      */
4507     insertBefore : function(el, o, returnElement){
4508         return this.doInsert(el, o, returnElement, "beforeBegin");
4509     },
4510
4511     /**
4512      * Creates new Dom element(s) and inserts them after el
4513      * @param {String/HTMLElement/Element} el The context element
4514      * @param {Object} o The Dom object spec (and children)
4515      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4516      * @return {HTMLElement/Roo.Element} The new node
4517      */
4518     insertAfter : function(el, o, returnElement){
4519         return this.doInsert(el, o, returnElement, "afterEnd", "nextSibling");
4520     },
4521
4522     /**
4523      * Creates new Dom element(s) and inserts them as the first child of el
4524      * @param {String/HTMLElement/Element} el The context element
4525      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4526      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4527      * @return {HTMLElement/Roo.Element} The new node
4528      */
4529     insertFirst : function(el, o, returnElement){
4530         return this.doInsert(el, o, returnElement, "afterBegin");
4531     },
4532
4533     // private
4534     doInsert : function(el, o, returnElement, pos, sibling){
4535         el = Roo.getDom(el);
4536         var newNode;
4537         if(this.useDom || o.ns){
4538             newNode = createDom(o, null);
4539             el.parentNode.insertBefore(newNode, sibling ? el[sibling] : el);
4540         }else{
4541             var html = createHtml(o);
4542             newNode = this.insertHtml(pos, el, html);
4543         }
4544         return returnElement ? Roo.get(newNode, true) : newNode;
4545     },
4546
4547     /**
4548      * Creates new Dom element(s) and appends them to el
4549      * @param {String/HTMLElement/Element} el The context element
4550      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4551      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4552      * @return {HTMLElement/Roo.Element} The new node
4553      */
4554     append : function(el, o, returnElement){
4555         el = Roo.getDom(el);
4556         var newNode;
4557         if(this.useDom || o.ns){
4558             newNode = createDom(o, null);
4559             el.appendChild(newNode);
4560         }else{
4561             var html = createHtml(o);
4562             newNode = this.insertHtml("beforeEnd", el, html);
4563         }
4564         return returnElement ? Roo.get(newNode, true) : newNode;
4565     },
4566
4567     /**
4568      * Creates new Dom element(s) and overwrites the contents of el with them
4569      * @param {String/HTMLElement/Element} el The context element
4570      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4571      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4572      * @return {HTMLElement/Roo.Element} The new node
4573      */
4574     overwrite : function(el, o, returnElement){
4575         el = Roo.getDom(el);
4576         if (o.ns) {
4577           
4578             while (el.childNodes.length) {
4579                 el.removeChild(el.firstChild);
4580             }
4581             createDom(o, el);
4582         } else {
4583             el.innerHTML = createHtml(o);   
4584         }
4585         
4586         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4587     },
4588
4589     /**
4590      * Creates a new Roo.DomHelper.Template from the Dom object spec
4591      * @param {Object} o The Dom object spec (and children)
4592      * @return {Roo.DomHelper.Template} The new template
4593      */
4594     createTemplate : function(o){
4595         var html = createHtml(o);
4596         return new Roo.Template(html);
4597     }
4598     };
4599 }();
4600 /*
4601  * Based on:
4602  * Ext JS Library 1.1.1
4603  * Copyright(c) 2006-2007, Ext JS, LLC.
4604  *
4605  * Originally Released Under LGPL - original licence link has changed is not relivant.
4606  *
4607  * Fork - LGPL
4608  * <script type="text/javascript">
4609  */
4610  
4611 /**
4612 * @class Roo.Template
4613 * Represents an HTML fragment template. Templates can be precompiled for greater performance.
4614 * For a list of available format functions, see {@link Roo.util.Format}.<br />
4615 * Usage:
4616 <pre><code>
4617 var t = new Roo.Template({
4618     html :  '&lt;div name="{id}"&gt;' + 
4619         '&lt;span class="{cls}"&gt;{name:trim} {someval:this.myformat}{value:ellipsis(10)}&lt;/span&gt;' +
4620         '&lt;/div&gt;',
4621     myformat: function (value, allValues) {
4622         return 'XX' + value;
4623     }
4624 });
4625 t.append('some-element', {id: 'myid', cls: 'myclass', name: 'foo', value: 'bar'});
4626 </code></pre>
4627 * For more information see this blog post with examples:
4628 *  <a href="http://www.cnitblog.com/seeyeah/archive/2011/12/30/38728.html/">DomHelper
4629      - Create Elements using DOM, HTML fragments and Templates</a>. 
4630 * @constructor
4631 * @param {Object} cfg - Configuration object.
4632 */
4633 Roo.Template = function(cfg){
4634     // BC!
4635     if(cfg instanceof Array){
4636         cfg = cfg.join("");
4637     }else if(arguments.length > 1){
4638         cfg = Array.prototype.join.call(arguments, "");
4639     }
4640     
4641     
4642     if (typeof(cfg) == 'object') {
4643         Roo.apply(this,cfg)
4644     } else {
4645         // bc
4646         this.html = cfg;
4647     }
4648     if (this.url) {
4649         this.load();
4650     }
4651     
4652 };
4653 Roo.Template.prototype = {
4654     
4655     /**
4656      * @cfg {String} url  The Url to load the template from. beware if you are loading from a url, the data may not be ready if you use it instantly..
4657      *                    it should be fixed so that template is observable...
4658      */
4659     url : false,
4660     /**
4661      * @cfg {String} html  The HTML fragment or an array of fragments to join("") or multiple arguments to join("")
4662      */
4663     html : '',
4664     /**
4665      * Returns an HTML fragment of this template with the specified values applied.
4666      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4667      * @return {String} The HTML fragment
4668      */
4669     applyTemplate : function(values){
4670         try {
4671            
4672             if(this.compiled){
4673                 return this.compiled(values);
4674             }
4675             var useF = this.disableFormats !== true;
4676             var fm = Roo.util.Format, tpl = this;
4677             var fn = function(m, name, format, args){
4678                 if(format && useF){
4679                     if(format.substr(0, 5) == "this."){
4680                         return tpl.call(format.substr(5), values[name], values);
4681                     }else{
4682                         if(args){
4683                             // quoted values are required for strings in compiled templates, 
4684                             // but for non compiled we need to strip them
4685                             // quoted reversed for jsmin
4686                             var re = /^\s*['"](.*)["']\s*$/;
4687                             args = args.split(',');
4688                             for(var i = 0, len = args.length; i < len; i++){
4689                                 args[i] = args[i].replace(re, "$1");
4690                             }
4691                             args = [values[name]].concat(args);
4692                         }else{
4693                             args = [values[name]];
4694                         }
4695                         return fm[format].apply(fm, args);
4696                     }
4697                 }else{
4698                     return values[name] !== undefined ? values[name] : "";
4699                 }
4700             };
4701             return this.html.replace(this.re, fn);
4702         } catch (e) {
4703             Roo.log(e);
4704             throw e;
4705         }
4706          
4707     },
4708     
4709     loading : false,
4710       
4711     load : function ()
4712     {
4713          
4714         if (this.loading) {
4715             return;
4716         }
4717         var _t = this;
4718         
4719         this.loading = true;
4720         this.compiled = false;
4721         
4722         var cx = new Roo.data.Connection();
4723         cx.request({
4724             url : this.url,
4725             method : 'GET',
4726             success : function (response) {
4727                 _t.loading = false;
4728                 _t.html = response.responseText;
4729                 _t.url = false;
4730                 _t.compile();
4731              },
4732             failure : function(response) {
4733                 Roo.log("Template failed to load from " + _t.url);
4734                 _t.loading = false;
4735             }
4736         });
4737     },
4738
4739     /**
4740      * Sets the HTML used as the template and optionally compiles it.
4741      * @param {String} html
4742      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4743      * @return {Roo.Template} this
4744      */
4745     set : function(html, compile){
4746         this.html = html;
4747         this.compiled = null;
4748         if(compile){
4749             this.compile();
4750         }
4751         return this;
4752     },
4753     
4754     /**
4755      * True to disable format functions (defaults to false)
4756      * @type Boolean
4757      */
4758     disableFormats : false,
4759     
4760     /**
4761     * The regular expression used to match template variables 
4762     * @type RegExp
4763     * @property 
4764     */
4765     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4766     
4767     /**
4768      * Compiles the template into an internal function, eliminating the RegEx overhead.
4769      * @return {Roo.Template} this
4770      */
4771     compile : function(){
4772         var fm = Roo.util.Format;
4773         var useF = this.disableFormats !== true;
4774         var sep = Roo.isGecko ? "+" : ",";
4775         var fn = function(m, name, format, args){
4776             if(format && useF){
4777                 args = args ? ',' + args : "";
4778                 if(format.substr(0, 5) != "this."){
4779                     format = "fm." + format + '(';
4780                 }else{
4781                     format = 'this.call("'+ format.substr(5) + '", ';
4782                     args = ", values";
4783                 }
4784             }else{
4785                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4786             }
4787             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4788         };
4789         var body;
4790         // branched to use + in gecko and [].join() in others
4791         if(Roo.isGecko){
4792             body = "this.compiled = function(values){ return '" +
4793                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4794                     "';};";
4795         }else{
4796             body = ["this.compiled = function(values){ return ['"];
4797             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4798             body.push("'].join('');};");
4799             body = body.join('');
4800         }
4801         /**
4802          * eval:var:values
4803          * eval:var:fm
4804          */
4805         eval(body);
4806         return this;
4807     },
4808     
4809     // private function used to call members
4810     call : function(fnName, value, allValues){
4811         return this[fnName](value, allValues);
4812     },
4813     
4814     /**
4815      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4816      * @param {String/HTMLElement/Roo.Element} el The context element
4817      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4818      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4819      * @return {HTMLElement/Roo.Element} The new node or Element
4820      */
4821     insertFirst: function(el, values, returnElement){
4822         return this.doInsert('afterBegin', el, values, returnElement);
4823     },
4824
4825     /**
4826      * Applies the supplied values to the template and inserts the new node(s) before el.
4827      * @param {String/HTMLElement/Roo.Element} el The context element
4828      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4829      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4830      * @return {HTMLElement/Roo.Element} The new node or Element
4831      */
4832     insertBefore: function(el, values, returnElement){
4833         return this.doInsert('beforeBegin', el, values, returnElement);
4834     },
4835
4836     /**
4837      * Applies the supplied values to the template and inserts the new node(s) after el.
4838      * @param {String/HTMLElement/Roo.Element} el The context element
4839      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4840      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4841      * @return {HTMLElement/Roo.Element} The new node or Element
4842      */
4843     insertAfter : function(el, values, returnElement){
4844         return this.doInsert('afterEnd', el, values, returnElement);
4845     },
4846     
4847     /**
4848      * Applies the supplied values to the template and appends the new node(s) to el.
4849      * @param {String/HTMLElement/Roo.Element} el The context element
4850      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4851      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4852      * @return {HTMLElement/Roo.Element} The new node or Element
4853      */
4854     append : function(el, values, returnElement){
4855         return this.doInsert('beforeEnd', el, values, returnElement);
4856     },
4857
4858     doInsert : function(where, el, values, returnEl){
4859         el = Roo.getDom(el);
4860         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4861         return returnEl ? Roo.get(newNode, true) : newNode;
4862     },
4863
4864     /**
4865      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4866      * @param {String/HTMLElement/Roo.Element} el The context element
4867      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4868      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4869      * @return {HTMLElement/Roo.Element} The new node or Element
4870      */
4871     overwrite : function(el, values, returnElement){
4872         el = Roo.getDom(el);
4873         el.innerHTML = this.applyTemplate(values);
4874         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4875     }
4876 };
4877 /**
4878  * Alias for {@link #applyTemplate}
4879  * @method
4880  */
4881 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4882
4883 // backwards compat
4884 Roo.DomHelper.Template = Roo.Template;
4885
4886 /**
4887  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4888  * @param {String/HTMLElement} el A DOM element or its id
4889  * @returns {Roo.Template} The created template
4890  * @static
4891  */
4892 Roo.Template.from = function(el){
4893     el = Roo.getDom(el);
4894     return new Roo.Template(el.value || el.innerHTML);
4895 };/*
4896  * Based on:
4897  * Ext JS Library 1.1.1
4898  * Copyright(c) 2006-2007, Ext JS, LLC.
4899  *
4900  * Originally Released Under LGPL - original licence link has changed is not relivant.
4901  *
4902  * Fork - LGPL
4903  * <script type="text/javascript">
4904  */
4905  
4906
4907 /*
4908  * This is code is also distributed under MIT license for use
4909  * with jQuery and prototype JavaScript libraries.
4910  */
4911 /**
4912  * @class Roo.DomQuery
4913 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4914 <p>
4915 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4916
4917 <p>
4918 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4919 </p>
4920 <h4>Element Selectors:</h4>
4921 <ul class="list">
4922     <li> <b>*</b> any element</li>
4923     <li> <b>E</b> an element with the tag E</li>
4924     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4925     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4926     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4927     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4928 </ul>
4929 <h4>Attribute Selectors:</h4>
4930 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4931 <ul class="list">
4932     <li> <b>E[foo]</b> has an attribute "foo"</li>
4933     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4934     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4935     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4936     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4937     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4938     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4939 </ul>
4940 <h4>Pseudo Classes:</h4>
4941 <ul class="list">
4942     <li> <b>E:first-child</b> E is the first child of its parent</li>
4943     <li> <b>E:last-child</b> E is the last child of its parent</li>
4944     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4945     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4946     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4947     <li> <b>E:only-child</b> E is the only child of its parent</li>
4948     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4949     <li> <b>E:first</b> the first E in the resultset</li>
4950     <li> <b>E:last</b> the last E in the resultset</li>
4951     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4952     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4953     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4954     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4955     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4956     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4957     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4958     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4959     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4960 </ul>
4961 <h4>CSS Value Selectors:</h4>
4962 <ul class="list">
4963     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4964     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4965     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4966     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4967     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4968     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4969 </ul>
4970  * @singleton
4971  */
4972 Roo.DomQuery = function(){
4973     var cache = {}, simpleCache = {}, valueCache = {};
4974     var nonSpace = /\S/;
4975     var trimRe = /^\s+|\s+$/g;
4976     var tplRe = /\{(\d+)\}/g;
4977     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4978     var tagTokenRe = /^(#)?([\w-\*]+)/;
4979     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4980
4981     function child(p, index){
4982         var i = 0;
4983         var n = p.firstChild;
4984         while(n){
4985             if(n.nodeType == 1){
4986                if(++i == index){
4987                    return n;
4988                }
4989             }
4990             n = n.nextSibling;
4991         }
4992         return null;
4993     };
4994
4995     function next(n){
4996         while((n = n.nextSibling) && n.nodeType != 1);
4997         return n;
4998     };
4999
5000     function prev(n){
5001         while((n = n.previousSibling) && n.nodeType != 1);
5002         return n;
5003     };
5004
5005     function children(d){
5006         var n = d.firstChild, ni = -1;
5007             while(n){
5008                 var nx = n.nextSibling;
5009                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5010                     d.removeChild(n);
5011                 }else{
5012                     n.nodeIndex = ++ni;
5013                 }
5014                 n = nx;
5015             }
5016             return this;
5017         };
5018
5019     function byClassName(c, a, v){
5020         if(!v){
5021             return c;
5022         }
5023         var r = [], ri = -1, cn;
5024         for(var i = 0, ci; ci = c[i]; i++){
5025             if((' '+ci.className+' ').indexOf(v) != -1){
5026                 r[++ri] = ci;
5027             }
5028         }
5029         return r;
5030     };
5031
5032     function attrValue(n, attr){
5033         if(!n.tagName && typeof n.length != "undefined"){
5034             n = n[0];
5035         }
5036         if(!n){
5037             return null;
5038         }
5039         if(attr == "for"){
5040             return n.htmlFor;
5041         }
5042         if(attr == "class" || attr == "className"){
5043             return n.className;
5044         }
5045         return n.getAttribute(attr) || n[attr];
5046
5047     };
5048
5049     function getNodes(ns, mode, tagName){
5050         var result = [], ri = -1, cs;
5051         if(!ns){
5052             return result;
5053         }
5054         tagName = tagName || "*";
5055         if(typeof ns.getElementsByTagName != "undefined"){
5056             ns = [ns];
5057         }
5058         if(!mode){
5059             for(var i = 0, ni; ni = ns[i]; i++){
5060                 cs = ni.getElementsByTagName(tagName);
5061                 for(var j = 0, ci; ci = cs[j]; j++){
5062                     result[++ri] = ci;
5063                 }
5064             }
5065         }else if(mode == "/" || mode == ">"){
5066             var utag = tagName.toUpperCase();
5067             for(var i = 0, ni, cn; ni = ns[i]; i++){
5068                 cn = ni.children || ni.childNodes;
5069                 for(var j = 0, cj; cj = cn[j]; j++){
5070                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5071                         result[++ri] = cj;
5072                     }
5073                 }
5074             }
5075         }else if(mode == "+"){
5076             var utag = tagName.toUpperCase();
5077             for(var i = 0, n; n = ns[i]; i++){
5078                 while((n = n.nextSibling) && n.nodeType != 1);
5079                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5080                     result[++ri] = n;
5081                 }
5082             }
5083         }else if(mode == "~"){
5084             for(var i = 0, n; n = ns[i]; i++){
5085                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5086                 if(n){
5087                     result[++ri] = n;
5088                 }
5089             }
5090         }
5091         return result;
5092     };
5093
5094     function concat(a, b){
5095         if(b.slice){
5096             return a.concat(b);
5097         }
5098         for(var i = 0, l = b.length; i < l; i++){
5099             a[a.length] = b[i];
5100         }
5101         return a;
5102     }
5103
5104     function byTag(cs, tagName){
5105         if(cs.tagName || cs == document){
5106             cs = [cs];
5107         }
5108         if(!tagName){
5109             return cs;
5110         }
5111         var r = [], ri = -1;
5112         tagName = tagName.toLowerCase();
5113         for(var i = 0, ci; ci = cs[i]; i++){
5114             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5115                 r[++ri] = ci;
5116             }
5117         }
5118         return r;
5119     };
5120
5121     function byId(cs, attr, id){
5122         if(cs.tagName || cs == document){
5123             cs = [cs];
5124         }
5125         if(!id){
5126             return cs;
5127         }
5128         var r = [], ri = -1;
5129         for(var i = 0,ci; ci = cs[i]; i++){
5130             if(ci && ci.id == id){
5131                 r[++ri] = ci;
5132                 return r;
5133             }
5134         }
5135         return r;
5136     };
5137
5138     function byAttribute(cs, attr, value, op, custom){
5139         var r = [], ri = -1, st = custom=="{";
5140         var f = Roo.DomQuery.operators[op];
5141         for(var i = 0, ci; ci = cs[i]; i++){
5142             var a;
5143             if(st){
5144                 a = Roo.DomQuery.getStyle(ci, attr);
5145             }
5146             else if(attr == "class" || attr == "className"){
5147                 a = ci.className;
5148             }else if(attr == "for"){
5149                 a = ci.htmlFor;
5150             }else if(attr == "href"){
5151                 a = ci.getAttribute("href", 2);
5152             }else{
5153                 a = ci.getAttribute(attr);
5154             }
5155             if((f && f(a, value)) || (!f && a)){
5156                 r[++ri] = ci;
5157             }
5158         }
5159         return r;
5160     };
5161
5162     function byPseudo(cs, name, value){
5163         return Roo.DomQuery.pseudos[name](cs, value);
5164     };
5165
5166     // This is for IE MSXML which does not support expandos.
5167     // IE runs the same speed using setAttribute, however FF slows way down
5168     // and Safari completely fails so they need to continue to use expandos.
5169     var isIE = window.ActiveXObject ? true : false;
5170
5171     // this eval is stop the compressor from
5172     // renaming the variable to something shorter
5173     
5174     /** eval:var:batch */
5175     var batch = 30803; 
5176
5177     var key = 30803;
5178
5179     function nodupIEXml(cs){
5180         var d = ++key;
5181         cs[0].setAttribute("_nodup", d);
5182         var r = [cs[0]];
5183         for(var i = 1, len = cs.length; i < len; i++){
5184             var c = cs[i];
5185             if(!c.getAttribute("_nodup") != d){
5186                 c.setAttribute("_nodup", d);
5187                 r[r.length] = c;
5188             }
5189         }
5190         for(var i = 0, len = cs.length; i < len; i++){
5191             cs[i].removeAttribute("_nodup");
5192         }
5193         return r;
5194     }
5195
5196     function nodup(cs){
5197         if(!cs){
5198             return [];
5199         }
5200         var len = cs.length, c, i, r = cs, cj, ri = -1;
5201         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5202             return cs;
5203         }
5204         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5205             return nodupIEXml(cs);
5206         }
5207         var d = ++key;
5208         cs[0]._nodup = d;
5209         for(i = 1; c = cs[i]; i++){
5210             if(c._nodup != d){
5211                 c._nodup = d;
5212             }else{
5213                 r = [];
5214                 for(var j = 0; j < i; j++){
5215                     r[++ri] = cs[j];
5216                 }
5217                 for(j = i+1; cj = cs[j]; j++){
5218                     if(cj._nodup != d){
5219                         cj._nodup = d;
5220                         r[++ri] = cj;
5221                     }
5222                 }
5223                 return r;
5224             }
5225         }
5226         return r;
5227     }
5228
5229     function quickDiffIEXml(c1, c2){
5230         var d = ++key;
5231         for(var i = 0, len = c1.length; i < len; i++){
5232             c1[i].setAttribute("_qdiff", d);
5233         }
5234         var r = [];
5235         for(var i = 0, len = c2.length; i < len; i++){
5236             if(c2[i].getAttribute("_qdiff") != d){
5237                 r[r.length] = c2[i];
5238             }
5239         }
5240         for(var i = 0, len = c1.length; i < len; i++){
5241            c1[i].removeAttribute("_qdiff");
5242         }
5243         return r;
5244     }
5245
5246     function quickDiff(c1, c2){
5247         var len1 = c1.length;
5248         if(!len1){
5249             return c2;
5250         }
5251         if(isIE && c1[0].selectSingleNode){
5252             return quickDiffIEXml(c1, c2);
5253         }
5254         var d = ++key;
5255         for(var i = 0; i < len1; i++){
5256             c1[i]._qdiff = d;
5257         }
5258         var r = [];
5259         for(var i = 0, len = c2.length; i < len; i++){
5260             if(c2[i]._qdiff != d){
5261                 r[r.length] = c2[i];
5262             }
5263         }
5264         return r;
5265     }
5266
5267     function quickId(ns, mode, root, id){
5268         if(ns == root){
5269            var d = root.ownerDocument || root;
5270            return d.getElementById(id);
5271         }
5272         ns = getNodes(ns, mode, "*");
5273         return byId(ns, null, id);
5274     }
5275
5276     return {
5277         getStyle : function(el, name){
5278             return Roo.fly(el).getStyle(name);
5279         },
5280         /**
5281          * Compiles a selector/xpath query into a reusable function. The returned function
5282          * takes one parameter "root" (optional), which is the context node from where the query should start.
5283          * @param {String} selector The selector/xpath query
5284          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5285          * @return {Function}
5286          */
5287         compile : function(path, type){
5288             type = type || "select";
5289             
5290             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5291             var q = path, mode, lq;
5292             var tk = Roo.DomQuery.matchers;
5293             var tklen = tk.length;
5294             var mm;
5295
5296             // accept leading mode switch
5297             var lmode = q.match(modeRe);
5298             if(lmode && lmode[1]){
5299                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5300                 q = q.replace(lmode[1], "");
5301             }
5302             // strip leading slashes
5303             while(path.substr(0, 1)=="/"){
5304                 path = path.substr(1);
5305             }
5306
5307             while(q && lq != q){
5308                 lq = q;
5309                 var tm = q.match(tagTokenRe);
5310                 if(type == "select"){
5311                     if(tm){
5312                         if(tm[1] == "#"){
5313                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5314                         }else{
5315                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5316                         }
5317                         q = q.replace(tm[0], "");
5318                     }else if(q.substr(0, 1) != '@'){
5319                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5320                     }
5321                 }else{
5322                     if(tm){
5323                         if(tm[1] == "#"){
5324                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5325                         }else{
5326                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5327                         }
5328                         q = q.replace(tm[0], "");
5329                     }
5330                 }
5331                 while(!(mm = q.match(modeRe))){
5332                     var matched = false;
5333                     for(var j = 0; j < tklen; j++){
5334                         var t = tk[j];
5335                         var m = q.match(t.re);
5336                         if(m){
5337                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5338                                                     return m[i];
5339                                                 });
5340                             q = q.replace(m[0], "");
5341                             matched = true;
5342                             break;
5343                         }
5344                     }
5345                     // prevent infinite loop on bad selector
5346                     if(!matched){
5347                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5348                     }
5349                 }
5350                 if(mm[1]){
5351                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5352                     q = q.replace(mm[1], "");
5353                 }
5354             }
5355             fn[fn.length] = "return nodup(n);\n}";
5356             
5357              /** 
5358               * list of variables that need from compression as they are used by eval.
5359              *  eval:var:batch 
5360              *  eval:var:nodup
5361              *  eval:var:byTag
5362              *  eval:var:ById
5363              *  eval:var:getNodes
5364              *  eval:var:quickId
5365              *  eval:var:mode
5366              *  eval:var:root
5367              *  eval:var:n
5368              *  eval:var:byClassName
5369              *  eval:var:byPseudo
5370              *  eval:var:byAttribute
5371              *  eval:var:attrValue
5372              * 
5373              **/ 
5374             eval(fn.join(""));
5375             return f;
5376         },
5377
5378         /**
5379          * Selects a group of elements.
5380          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5381          * @param {Node} root (optional) The start of the query (defaults to document).
5382          * @return {Array}
5383          */
5384         select : function(path, root, type){
5385             if(!root || root == document){
5386                 root = document;
5387             }
5388             if(typeof root == "string"){
5389                 root = document.getElementById(root);
5390             }
5391             var paths = path.split(",");
5392             var results = [];
5393             for(var i = 0, len = paths.length; i < len; i++){
5394                 var p = paths[i].replace(trimRe, "");
5395                 if(!cache[p]){
5396                     cache[p] = Roo.DomQuery.compile(p);
5397                     if(!cache[p]){
5398                         throw p + " is not a valid selector";
5399                     }
5400                 }
5401                 var result = cache[p](root);
5402                 if(result && result != document){
5403                     results = results.concat(result);
5404                 }
5405             }
5406             if(paths.length > 1){
5407                 return nodup(results);
5408             }
5409             return results;
5410         },
5411
5412         /**
5413          * Selects a single element.
5414          * @param {String} selector The selector/xpath query
5415          * @param {Node} root (optional) The start of the query (defaults to document).
5416          * @return {Element}
5417          */
5418         selectNode : function(path, root){
5419             return Roo.DomQuery.select(path, root)[0];
5420         },
5421
5422         /**
5423          * Selects the value of a node, optionally replacing null with the defaultValue.
5424          * @param {String} selector The selector/xpath query
5425          * @param {Node} root (optional) The start of the query (defaults to document).
5426          * @param {String} defaultValue
5427          */
5428         selectValue : function(path, root, defaultValue){
5429             path = path.replace(trimRe, "");
5430             if(!valueCache[path]){
5431                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5432             }
5433             var n = valueCache[path](root);
5434             n = n[0] ? n[0] : n;
5435             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5436             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5437         },
5438
5439         /**
5440          * Selects the value of a node, parsing integers and floats.
5441          * @param {String} selector The selector/xpath query
5442          * @param {Node} root (optional) The start of the query (defaults to document).
5443          * @param {Number} defaultValue
5444          * @return {Number}
5445          */
5446         selectNumber : function(path, root, defaultValue){
5447             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5448             return parseFloat(v);
5449         },
5450
5451         /**
5452          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5453          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5454          * @param {String} selector The simple selector to test
5455          * @return {Boolean}
5456          */
5457         is : function(el, ss){
5458             if(typeof el == "string"){
5459                 el = document.getElementById(el);
5460             }
5461             var isArray = (el instanceof Array);
5462             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5463             return isArray ? (result.length == el.length) : (result.length > 0);
5464         },
5465
5466         /**
5467          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5468          * @param {Array} el An array of elements to filter
5469          * @param {String} selector The simple selector to test
5470          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5471          * the selector instead of the ones that match
5472          * @return {Array}
5473          */
5474         filter : function(els, ss, nonMatches){
5475             ss = ss.replace(trimRe, "");
5476             if(!simpleCache[ss]){
5477                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5478             }
5479             var result = simpleCache[ss](els);
5480             return nonMatches ? quickDiff(result, els) : result;
5481         },
5482
5483         /**
5484          * Collection of matching regular expressions and code snippets.
5485          */
5486         matchers : [{
5487                 re: /^\.([\w-]+)/,
5488                 select: 'n = byClassName(n, null, " {1} ");'
5489             }, {
5490                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5491                 select: 'n = byPseudo(n, "{1}", "{2}");'
5492             },{
5493                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5494                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5495             }, {
5496                 re: /^#([\w-]+)/,
5497                 select: 'n = byId(n, null, "{1}");'
5498             },{
5499                 re: /^@([\w-]+)/,
5500                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5501             }
5502         ],
5503
5504         /**
5505          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5506          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5507          */
5508         operators : {
5509             "=" : function(a, v){
5510                 return a == v;
5511             },
5512             "!=" : function(a, v){
5513                 return a != v;
5514             },
5515             "^=" : function(a, v){
5516                 return a && a.substr(0, v.length) == v;
5517             },
5518             "$=" : function(a, v){
5519                 return a && a.substr(a.length-v.length) == v;
5520             },
5521             "*=" : function(a, v){
5522                 return a && a.indexOf(v) !== -1;
5523             },
5524             "%=" : function(a, v){
5525                 return (a % v) == 0;
5526             },
5527             "|=" : function(a, v){
5528                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5529             },
5530             "~=" : function(a, v){
5531                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5532             }
5533         },
5534
5535         /**
5536          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5537          * and the argument (if any) supplied in the selector.
5538          */
5539         pseudos : {
5540             "first-child" : function(c){
5541                 var r = [], ri = -1, n;
5542                 for(var i = 0, ci; ci = n = c[i]; i++){
5543                     while((n = n.previousSibling) && n.nodeType != 1);
5544                     if(!n){
5545                         r[++ri] = ci;
5546                     }
5547                 }
5548                 return r;
5549             },
5550
5551             "last-child" : function(c){
5552                 var r = [], ri = -1, n;
5553                 for(var i = 0, ci; ci = n = c[i]; i++){
5554                     while((n = n.nextSibling) && n.nodeType != 1);
5555                     if(!n){
5556                         r[++ri] = ci;
5557                     }
5558                 }
5559                 return r;
5560             },
5561
5562             "nth-child" : function(c, a) {
5563                 var r = [], ri = -1;
5564                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5565                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5566                 for(var i = 0, n; n = c[i]; i++){
5567                     var pn = n.parentNode;
5568                     if (batch != pn._batch) {
5569                         var j = 0;
5570                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5571                             if(cn.nodeType == 1){
5572                                cn.nodeIndex = ++j;
5573                             }
5574                         }
5575                         pn._batch = batch;
5576                     }
5577                     if (f == 1) {
5578                         if (l == 0 || n.nodeIndex == l){
5579                             r[++ri] = n;
5580                         }
5581                     } else if ((n.nodeIndex + l) % f == 0){
5582                         r[++ri] = n;
5583                     }
5584                 }
5585
5586                 return r;
5587             },
5588
5589             "only-child" : function(c){
5590                 var r = [], ri = -1;;
5591                 for(var i = 0, ci; ci = c[i]; i++){
5592                     if(!prev(ci) && !next(ci)){
5593                         r[++ri] = ci;
5594                     }
5595                 }
5596                 return r;
5597             },
5598
5599             "empty" : function(c){
5600                 var r = [], ri = -1;
5601                 for(var i = 0, ci; ci = c[i]; i++){
5602                     var cns = ci.childNodes, j = 0, cn, empty = true;
5603                     while(cn = cns[j]){
5604                         ++j;
5605                         if(cn.nodeType == 1 || cn.nodeType == 3){
5606                             empty = false;
5607                             break;
5608                         }
5609                     }
5610                     if(empty){
5611                         r[++ri] = ci;
5612                     }
5613                 }
5614                 return r;
5615             },
5616
5617             "contains" : function(c, v){
5618                 var r = [], ri = -1;
5619                 for(var i = 0, ci; ci = c[i]; i++){
5620                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5621                         r[++ri] = ci;
5622                     }
5623                 }
5624                 return r;
5625             },
5626
5627             "nodeValue" : function(c, v){
5628                 var r = [], ri = -1;
5629                 for(var i = 0, ci; ci = c[i]; i++){
5630                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5631                         r[++ri] = ci;
5632                     }
5633                 }
5634                 return r;
5635             },
5636
5637             "checked" : function(c){
5638                 var r = [], ri = -1;
5639                 for(var i = 0, ci; ci = c[i]; i++){
5640                     if(ci.checked == true){
5641                         r[++ri] = ci;
5642                     }
5643                 }
5644                 return r;
5645             },
5646
5647             "not" : function(c, ss){
5648                 return Roo.DomQuery.filter(c, ss, true);
5649             },
5650
5651             "odd" : function(c){
5652                 return this["nth-child"](c, "odd");
5653             },
5654
5655             "even" : function(c){
5656                 return this["nth-child"](c, "even");
5657             },
5658
5659             "nth" : function(c, a){
5660                 return c[a-1] || [];
5661             },
5662
5663             "first" : function(c){
5664                 return c[0] || [];
5665             },
5666
5667             "last" : function(c){
5668                 return c[c.length-1] || [];
5669             },
5670
5671             "has" : function(c, ss){
5672                 var s = Roo.DomQuery.select;
5673                 var r = [], ri = -1;
5674                 for(var i = 0, ci; ci = c[i]; i++){
5675                     if(s(ss, ci).length > 0){
5676                         r[++ri] = ci;
5677                     }
5678                 }
5679                 return r;
5680             },
5681
5682             "next" : function(c, ss){
5683                 var is = Roo.DomQuery.is;
5684                 var r = [], ri = -1;
5685                 for(var i = 0, ci; ci = c[i]; i++){
5686                     var n = next(ci);
5687                     if(n && is(n, ss)){
5688                         r[++ri] = ci;
5689                     }
5690                 }
5691                 return r;
5692             },
5693
5694             "prev" : function(c, ss){
5695                 var is = Roo.DomQuery.is;
5696                 var r = [], ri = -1;
5697                 for(var i = 0, ci; ci = c[i]; i++){
5698                     var n = prev(ci);
5699                     if(n && is(n, ss)){
5700                         r[++ri] = ci;
5701                     }
5702                 }
5703                 return r;
5704             }
5705         }
5706     };
5707 }();
5708
5709 /**
5710  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5711  * @param {String} path The selector/xpath query
5712  * @param {Node} root (optional) The start of the query (defaults to document).
5713  * @return {Array}
5714  * @member Roo
5715  * @method query
5716  */
5717 Roo.query = Roo.DomQuery.select;
5718 /*
5719  * Based on:
5720  * Ext JS Library 1.1.1
5721  * Copyright(c) 2006-2007, Ext JS, LLC.
5722  *
5723  * Originally Released Under LGPL - original licence link has changed is not relivant.
5724  *
5725  * Fork - LGPL
5726  * <script type="text/javascript">
5727  */
5728
5729 /**
5730  * @class Roo.util.Observable
5731  * Base class that provides a common interface for publishing events. Subclasses are expected to
5732  * to have a property "events" with all the events defined.<br>
5733  * For example:
5734  * <pre><code>
5735  Employee = function(name){
5736     this.name = name;
5737     this.addEvents({
5738         "fired" : true,
5739         "quit" : true
5740     });
5741  }
5742  Roo.extend(Employee, Roo.util.Observable);
5743 </code></pre>
5744  * @param {Object} config properties to use (incuding events / listeners)
5745  */
5746
5747 Roo.util.Observable = function(cfg){
5748     
5749     cfg = cfg|| {};
5750     this.addEvents(cfg.events || {});
5751     if (cfg.events) {
5752         delete cfg.events; // make sure
5753     }
5754      
5755     Roo.apply(this, cfg);
5756     
5757     if(this.listeners){
5758         this.on(this.listeners);
5759         delete this.listeners;
5760     }
5761 };
5762 Roo.util.Observable.prototype = {
5763     /** 
5764  * @cfg {Object} listeners  list of events and functions to call for this object, 
5765  * For example :
5766  * <pre><code>
5767     listeners :  { 
5768        'click' : function(e) {
5769            ..... 
5770         } ,
5771         .... 
5772     } 
5773   </code></pre>
5774  */
5775     
5776     
5777     /**
5778      * Fires the specified event with the passed parameters (minus the event name).
5779      * @param {String} eventName
5780      * @param {Object...} args Variable number of parameters are passed to handlers
5781      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5782      */
5783     fireEvent : function(){
5784         var ce = this.events[arguments[0].toLowerCase()];
5785         if(typeof ce == "object"){
5786             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5787         }else{
5788             return true;
5789         }
5790     },
5791
5792     // private
5793     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5794
5795     /**
5796      * Appends an event handler to this component
5797      * @param {String}   eventName The type of event to listen for
5798      * @param {Function} handler The method the event invokes
5799      * @param {Object}   scope (optional) The scope in which to execute the handler
5800      * function. The handler function's "this" context.
5801      * @param {Object}   options (optional) An object containing handler configuration
5802      * properties. This may contain any of the following properties:<ul>
5803      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5804      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5805      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5806      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5807      * by the specified number of milliseconds. If the event fires again within that time, the original
5808      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5809      * </ul><br>
5810      * <p>
5811      * <b>Combining Options</b><br>
5812      * Using the options argument, it is possible to combine different types of listeners:<br>
5813      * <br>
5814      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5815                 <pre><code>
5816                 el.on('click', this.onClick, this, {
5817                         single: true,
5818                 delay: 100,
5819                 forumId: 4
5820                 });
5821                 </code></pre>
5822      * <p>
5823      * <b>Attaching multiple handlers in 1 call</b><br>
5824      * The method also allows for a single argument to be passed which is a config object containing properties
5825      * which specify multiple handlers.
5826      * <pre><code>
5827                 el.on({
5828                         'click': {
5829                         fn: this.onClick,
5830                         scope: this,
5831                         delay: 100
5832                 }, 
5833                 'mouseover': {
5834                         fn: this.onMouseOver,
5835                         scope: this
5836                 },
5837                 'mouseout': {
5838                         fn: this.onMouseOut,
5839                         scope: this
5840                 }
5841                 });
5842                 </code></pre>
5843      * <p>
5844      * Or a shorthand syntax which passes the same scope object to all handlers:
5845         <pre><code>
5846                 el.on({
5847                         'click': this.onClick,
5848                 'mouseover': this.onMouseOver,
5849                 'mouseout': this.onMouseOut,
5850                 scope: this
5851                 });
5852                 </code></pre>
5853      */
5854     addListener : function(eventName, fn, scope, o){
5855         if(typeof eventName == "object"){
5856             o = eventName;
5857             for(var e in o){
5858                 if(this.filterOptRe.test(e)){
5859                     continue;
5860                 }
5861                 if(typeof o[e] == "function"){
5862                     // shared options
5863                     this.addListener(e, o[e], o.scope,  o);
5864                 }else{
5865                     // individual options
5866                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5867                 }
5868             }
5869             return;
5870         }
5871         o = (!o || typeof o == "boolean") ? {} : o;
5872         eventName = eventName.toLowerCase();
5873         var ce = this.events[eventName] || true;
5874         if(typeof ce == "boolean"){
5875             ce = new Roo.util.Event(this, eventName);
5876             this.events[eventName] = ce;
5877         }
5878         ce.addListener(fn, scope, o);
5879     },
5880
5881     /**
5882      * Removes a listener
5883      * @param {String}   eventName     The type of event to listen for
5884      * @param {Function} handler        The handler to remove
5885      * @param {Object}   scope  (optional) The scope (this object) for the handler
5886      */
5887     removeListener : function(eventName, fn, scope){
5888         var ce = this.events[eventName.toLowerCase()];
5889         if(typeof ce == "object"){
5890             ce.removeListener(fn, scope);
5891         }
5892     },
5893
5894     /**
5895      * Removes all listeners for this object
5896      */
5897     purgeListeners : function(){
5898         for(var evt in this.events){
5899             if(typeof this.events[evt] == "object"){
5900                  this.events[evt].clearListeners();
5901             }
5902         }
5903     },
5904
5905     relayEvents : function(o, events){
5906         var createHandler = function(ename){
5907             return function(){
5908                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5909             };
5910         };
5911         for(var i = 0, len = events.length; i < len; i++){
5912             var ename = events[i];
5913             if(!this.events[ename]){ this.events[ename] = true; };
5914             o.on(ename, createHandler(ename), this);
5915         }
5916     },
5917
5918     /**
5919      * Used to define events on this Observable
5920      * @param {Object} object The object with the events defined
5921      */
5922     addEvents : function(o){
5923         if(!this.events){
5924             this.events = {};
5925         }
5926         Roo.applyIf(this.events, o);
5927     },
5928
5929     /**
5930      * Checks to see if this object has any listeners for a specified event
5931      * @param {String} eventName The name of the event to check for
5932      * @return {Boolean} True if the event is being listened for, else false
5933      */
5934     hasListener : function(eventName){
5935         var e = this.events[eventName];
5936         return typeof e == "object" && e.listeners.length > 0;
5937     }
5938 };
5939 /**
5940  * Appends an event handler to this element (shorthand for addListener)
5941  * @param {String}   eventName     The type of event to listen for
5942  * @param {Function} handler        The method the event invokes
5943  * @param {Object}   scope (optional) The scope in which to execute the handler
5944  * function. The handler function's "this" context.
5945  * @param {Object}   options  (optional)
5946  * @method
5947  */
5948 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5949 /**
5950  * Removes a listener (shorthand for removeListener)
5951  * @param {String}   eventName     The type of event to listen for
5952  * @param {Function} handler        The handler to remove
5953  * @param {Object}   scope  (optional) The scope (this object) for the handler
5954  * @method
5955  */
5956 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5957
5958 /**
5959  * Starts capture on the specified Observable. All events will be passed
5960  * to the supplied function with the event name + standard signature of the event
5961  * <b>before</b> the event is fired. If the supplied function returns false,
5962  * the event will not fire.
5963  * @param {Observable} o The Observable to capture
5964  * @param {Function} fn The function to call
5965  * @param {Object} scope (optional) The scope (this object) for the fn
5966  * @static
5967  */
5968 Roo.util.Observable.capture = function(o, fn, scope){
5969     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5970 };
5971
5972 /**
5973  * Removes <b>all</b> added captures from the Observable.
5974  * @param {Observable} o The Observable to release
5975  * @static
5976  */
5977 Roo.util.Observable.releaseCapture = function(o){
5978     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5979 };
5980
5981 (function(){
5982
5983     var createBuffered = function(h, o, scope){
5984         var task = new Roo.util.DelayedTask();
5985         return function(){
5986             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5987         };
5988     };
5989
5990     var createSingle = function(h, e, fn, scope){
5991         return function(){
5992             e.removeListener(fn, scope);
5993             return h.apply(scope, arguments);
5994         };
5995     };
5996
5997     var createDelayed = function(h, o, scope){
5998         return function(){
5999             var args = Array.prototype.slice.call(arguments, 0);
6000             setTimeout(function(){
6001                 h.apply(scope, args);
6002             }, o.delay || 10);
6003         };
6004     };
6005
6006     Roo.util.Event = function(obj, name){
6007         this.name = name;
6008         this.obj = obj;
6009         this.listeners = [];
6010     };
6011
6012     Roo.util.Event.prototype = {
6013         addListener : function(fn, scope, options){
6014             var o = options || {};
6015             scope = scope || this.obj;
6016             if(!this.isListening(fn, scope)){
6017                 var l = {fn: fn, scope: scope, options: o};
6018                 var h = fn;
6019                 if(o.delay){
6020                     h = createDelayed(h, o, scope);
6021                 }
6022                 if(o.single){
6023                     h = createSingle(h, this, fn, scope);
6024                 }
6025                 if(o.buffer){
6026                     h = createBuffered(h, o, scope);
6027                 }
6028                 l.fireFn = h;
6029                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6030                     this.listeners.push(l);
6031                 }else{
6032                     this.listeners = this.listeners.slice(0);
6033                     this.listeners.push(l);
6034                 }
6035             }
6036         },
6037
6038         findListener : function(fn, scope){
6039             scope = scope || this.obj;
6040             var ls = this.listeners;
6041             for(var i = 0, len = ls.length; i < len; i++){
6042                 var l = ls[i];
6043                 if(l.fn == fn && l.scope == scope){
6044                     return i;
6045                 }
6046             }
6047             return -1;
6048         },
6049
6050         isListening : function(fn, scope){
6051             return this.findListener(fn, scope) != -1;
6052         },
6053
6054         removeListener : function(fn, scope){
6055             var index;
6056             if((index = this.findListener(fn, scope)) != -1){
6057                 if(!this.firing){
6058                     this.listeners.splice(index, 1);
6059                 }else{
6060                     this.listeners = this.listeners.slice(0);
6061                     this.listeners.splice(index, 1);
6062                 }
6063                 return true;
6064             }
6065             return false;
6066         },
6067
6068         clearListeners : function(){
6069             this.listeners = [];
6070         },
6071
6072         fire : function(){
6073             var ls = this.listeners, scope, len = ls.length;
6074             if(len > 0){
6075                 this.firing = true;
6076                 var args = Array.prototype.slice.call(arguments, 0);
6077                 for(var i = 0; i < len; i++){
6078                     var l = ls[i];
6079                     if(l.fireFn.apply(l.scope||this.obj||window, arguments) === false){
6080                         this.firing = false;
6081                         return false;
6082                     }
6083                 }
6084                 this.firing = false;
6085             }
6086             return true;
6087         }
6088     };
6089 })();/*
6090  * RooJS Library 
6091  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6092  *
6093  * Licence LGPL 
6094  *
6095  */
6096  
6097 /**
6098  * @class Roo.Document
6099  * @extends Roo.util.Observable
6100  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6101  * 
6102  * @param {Object} config the methods and properties of the 'base' class for the application.
6103  * 
6104  *  Generic Page handler - implement this to start your app..
6105  * 
6106  * eg.
6107  *  MyProject = new Roo.Document({
6108         events : {
6109             'load' : true // your events..
6110         },
6111         listeners : {
6112             'ready' : function() {
6113                 // fired on Roo.onReady()
6114             }
6115         }
6116  * 
6117  */
6118 Roo.Document = function(cfg) {
6119      
6120     this.addEvents({ 
6121         'ready' : true
6122     });
6123     Roo.util.Observable.call(this,cfg);
6124     
6125     var _this = this;
6126     
6127     Roo.onReady(function() {
6128         _this.fireEvent('ready');
6129     },null,false);
6130     
6131     
6132 }
6133
6134 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6135  * Based on:
6136  * Ext JS Library 1.1.1
6137  * Copyright(c) 2006-2007, Ext JS, LLC.
6138  *
6139  * Originally Released Under LGPL - original licence link has changed is not relivant.
6140  *
6141  * Fork - LGPL
6142  * <script type="text/javascript">
6143  */
6144
6145 /**
6146  * @class Roo.EventManager
6147  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6148  * several useful events directly.
6149  * See {@link Roo.EventObject} for more details on normalized event objects.
6150  * @singleton
6151  */
6152 Roo.EventManager = function(){
6153     var docReadyEvent, docReadyProcId, docReadyState = false;
6154     var resizeEvent, resizeTask, textEvent, textSize;
6155     var E = Roo.lib.Event;
6156     var D = Roo.lib.Dom;
6157
6158     
6159     
6160
6161     var fireDocReady = function(){
6162         if(!docReadyState){
6163             docReadyState = true;
6164             Roo.isReady = true;
6165             if(docReadyProcId){
6166                 clearInterval(docReadyProcId);
6167             }
6168             if(Roo.isGecko || Roo.isOpera) {
6169                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6170             }
6171             if(Roo.isIE){
6172                 var defer = document.getElementById("ie-deferred-loader");
6173                 if(defer){
6174                     defer.onreadystatechange = null;
6175                     defer.parentNode.removeChild(defer);
6176                 }
6177             }
6178             if(docReadyEvent){
6179                 docReadyEvent.fire();
6180                 docReadyEvent.clearListeners();
6181             }
6182         }
6183     };
6184     
6185     var initDocReady = function(){
6186         docReadyEvent = new Roo.util.Event();
6187         if(Roo.isGecko || Roo.isOpera) {
6188             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6189         }else if(Roo.isIE){
6190             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6191             var defer = document.getElementById("ie-deferred-loader");
6192             defer.onreadystatechange = function(){
6193                 if(this.readyState == "complete"){
6194                     fireDocReady();
6195                 }
6196             };
6197         }else if(Roo.isSafari){ 
6198             docReadyProcId = setInterval(function(){
6199                 var rs = document.readyState;
6200                 if(rs == "complete") {
6201                     fireDocReady();     
6202                  }
6203             }, 10);
6204         }
6205         // no matter what, make sure it fires on load
6206         E.on(window, "load", fireDocReady);
6207     };
6208
6209     var createBuffered = function(h, o){
6210         var task = new Roo.util.DelayedTask(h);
6211         return function(e){
6212             // create new event object impl so new events don't wipe out properties
6213             e = new Roo.EventObjectImpl(e);
6214             task.delay(o.buffer, h, null, [e]);
6215         };
6216     };
6217
6218     var createSingle = function(h, el, ename, fn){
6219         return function(e){
6220             Roo.EventManager.removeListener(el, ename, fn);
6221             h(e);
6222         };
6223     };
6224
6225     var createDelayed = function(h, o){
6226         return function(e){
6227             // create new event object impl so new events don't wipe out properties
6228             e = new Roo.EventObjectImpl(e);
6229             setTimeout(function(){
6230                 h(e);
6231             }, o.delay || 10);
6232         };
6233     };
6234     var transitionEndVal = false;
6235     
6236     var transitionEnd = function()
6237     {
6238         if (transitionEndVal) {
6239             return transitionEndVal;
6240         }
6241         var el = document.createElement('div');
6242
6243         var transEndEventNames = {
6244             WebkitTransition : 'webkitTransitionEnd',
6245             MozTransition    : 'transitionend',
6246             OTransition      : 'oTransitionEnd otransitionend',
6247             transition       : 'transitionend'
6248         };
6249     
6250         for (var name in transEndEventNames) {
6251             if (el.style[name] !== undefined) {
6252                 transitionEndVal = transEndEventNames[name];
6253                 return  transitionEndVal ;
6254             }
6255         }
6256     }
6257     
6258
6259     var listen = function(element, ename, opt, fn, scope){
6260         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6261         fn = fn || o.fn; scope = scope || o.scope;
6262         var el = Roo.getDom(element);
6263         
6264         
6265         if(!el){
6266             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6267         }
6268         
6269         if (ename == 'transitionend') {
6270             ename = transitionEnd();
6271         }
6272         var h = function(e){
6273             e = Roo.EventObject.setEvent(e);
6274             var t;
6275             if(o.delegate){
6276                 t = e.getTarget(o.delegate, el);
6277                 if(!t){
6278                     return;
6279                 }
6280             }else{
6281                 t = e.target;
6282             }
6283             if(o.stopEvent === true){
6284                 e.stopEvent();
6285             }
6286             if(o.preventDefault === true){
6287                e.preventDefault();
6288             }
6289             if(o.stopPropagation === true){
6290                 e.stopPropagation();
6291             }
6292
6293             if(o.normalized === false){
6294                 e = e.browserEvent;
6295             }
6296
6297             fn.call(scope || el, e, t, o);
6298         };
6299         if(o.delay){
6300             h = createDelayed(h, o);
6301         }
6302         if(o.single){
6303             h = createSingle(h, el, ename, fn);
6304         }
6305         if(o.buffer){
6306             h = createBuffered(h, o);
6307         }
6308         
6309         fn._handlers = fn._handlers || [];
6310         
6311         
6312         fn._handlers.push([Roo.id(el), ename, h]);
6313         
6314         
6315          
6316         E.on(el, ename, h);
6317         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6318             el.addEventListener("DOMMouseScroll", h, false);
6319             E.on(window, 'unload', function(){
6320                 el.removeEventListener("DOMMouseScroll", h, false);
6321             });
6322         }
6323         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6324             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6325         }
6326         return h;
6327     };
6328
6329     var stopListening = function(el, ename, fn){
6330         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6331         if(hds){
6332             for(var i = 0, len = hds.length; i < len; i++){
6333                 var h = hds[i];
6334                 if(h[0] == id && h[1] == ename){
6335                     hd = h[2];
6336                     hds.splice(i, 1);
6337                     break;
6338                 }
6339             }
6340         }
6341         E.un(el, ename, hd);
6342         el = Roo.getDom(el);
6343         if(ename == "mousewheel" && el.addEventListener){
6344             el.removeEventListener("DOMMouseScroll", hd, false);
6345         }
6346         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6347             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6348         }
6349     };
6350
6351     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6352     
6353     var pub = {
6354         
6355         
6356         /** 
6357          * Fix for doc tools
6358          * @scope Roo.EventManager
6359          */
6360         
6361         
6362         /** 
6363          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6364          * object with a Roo.EventObject
6365          * @param {Function} fn        The method the event invokes
6366          * @param {Object}   scope    An object that becomes the scope of the handler
6367          * @param {boolean}  override If true, the obj passed in becomes
6368          *                             the execution scope of the listener
6369          * @return {Function} The wrapped function
6370          * @deprecated
6371          */
6372         wrap : function(fn, scope, override){
6373             return function(e){
6374                 Roo.EventObject.setEvent(e);
6375                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6376             };
6377         },
6378         
6379         /**
6380      * Appends an event handler to an element (shorthand for addListener)
6381      * @param {String/HTMLElement}   element        The html element or id to assign the
6382      * @param {String}   eventName The type of event to listen for
6383      * @param {Function} handler The method the event invokes
6384      * @param {Object}   scope (optional) The scope in which to execute the handler
6385      * function. The handler function's "this" context.
6386      * @param {Object}   options (optional) An object containing handler configuration
6387      * properties. This may contain any of the following properties:<ul>
6388      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6389      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6390      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6391      * <li>preventDefault {Boolean} True to prevent the default action</li>
6392      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6393      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6394      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6395      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6396      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6397      * by the specified number of milliseconds. If the event fires again within that time, the original
6398      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6399      * </ul><br>
6400      * <p>
6401      * <b>Combining Options</b><br>
6402      * Using the options argument, it is possible to combine different types of listeners:<br>
6403      * <br>
6404      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6405      * Code:<pre><code>
6406 el.on('click', this.onClick, this, {
6407     single: true,
6408     delay: 100,
6409     stopEvent : true,
6410     forumId: 4
6411 });</code></pre>
6412      * <p>
6413      * <b>Attaching multiple handlers in 1 call</b><br>
6414       * The method also allows for a single argument to be passed which is a config object containing properties
6415      * which specify multiple handlers.
6416      * <p>
6417      * Code:<pre><code>
6418 el.on({
6419     'click' : {
6420         fn: this.onClick
6421         scope: this,
6422         delay: 100
6423     },
6424     'mouseover' : {
6425         fn: this.onMouseOver
6426         scope: this
6427     },
6428     'mouseout' : {
6429         fn: this.onMouseOut
6430         scope: this
6431     }
6432 });</code></pre>
6433      * <p>
6434      * Or a shorthand syntax:<br>
6435      * Code:<pre><code>
6436 el.on({
6437     'click' : this.onClick,
6438     'mouseover' : this.onMouseOver,
6439     'mouseout' : this.onMouseOut
6440     scope: this
6441 });</code></pre>
6442      */
6443         addListener : function(element, eventName, fn, scope, options){
6444             if(typeof eventName == "object"){
6445                 var o = eventName;
6446                 for(var e in o){
6447                     if(propRe.test(e)){
6448                         continue;
6449                     }
6450                     if(typeof o[e] == "function"){
6451                         // shared options
6452                         listen(element, e, o, o[e], o.scope);
6453                     }else{
6454                         // individual options
6455                         listen(element, e, o[e]);
6456                     }
6457                 }
6458                 return;
6459             }
6460             return listen(element, eventName, options, fn, scope);
6461         },
6462         
6463         /**
6464          * Removes an event handler
6465          *
6466          * @param {String/HTMLElement}   element        The id or html element to remove the 
6467          *                             event from
6468          * @param {String}   eventName     The type of event
6469          * @param {Function} fn
6470          * @return {Boolean} True if a listener was actually removed
6471          */
6472         removeListener : function(element, eventName, fn){
6473             return stopListening(element, eventName, fn);
6474         },
6475         
6476         /**
6477          * Fires when the document is ready (before onload and before images are loaded). Can be 
6478          * accessed shorthanded Roo.onReady().
6479          * @param {Function} fn        The method the event invokes
6480          * @param {Object}   scope    An  object that becomes the scope of the handler
6481          * @param {boolean}  options
6482          */
6483         onDocumentReady : function(fn, scope, options){
6484             if(docReadyState){ // if it already fired
6485                 docReadyEvent.addListener(fn, scope, options);
6486                 docReadyEvent.fire();
6487                 docReadyEvent.clearListeners();
6488                 return;
6489             }
6490             if(!docReadyEvent){
6491                 initDocReady();
6492             }
6493             docReadyEvent.addListener(fn, scope, options);
6494         },
6495         
6496         /**
6497          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6498          * @param {Function} fn        The method the event invokes
6499          * @param {Object}   scope    An object that becomes the scope of the handler
6500          * @param {boolean}  options
6501          */
6502         onWindowResize : function(fn, scope, options){
6503             if(!resizeEvent){
6504                 resizeEvent = new Roo.util.Event();
6505                 resizeTask = new Roo.util.DelayedTask(function(){
6506                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6507                 });
6508                 E.on(window, "resize", function(){
6509                     if(Roo.isIE){
6510                         resizeTask.delay(50);
6511                     }else{
6512                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6513                     }
6514                 });
6515             }
6516             resizeEvent.addListener(fn, scope, options);
6517         },
6518
6519         /**
6520          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6521          * @param {Function} fn        The method the event invokes
6522          * @param {Object}   scope    An object that becomes the scope of the handler
6523          * @param {boolean}  options
6524          */
6525         onTextResize : function(fn, scope, options){
6526             if(!textEvent){
6527                 textEvent = new Roo.util.Event();
6528                 var textEl = new Roo.Element(document.createElement('div'));
6529                 textEl.dom.className = 'x-text-resize';
6530                 textEl.dom.innerHTML = 'X';
6531                 textEl.appendTo(document.body);
6532                 textSize = textEl.dom.offsetHeight;
6533                 setInterval(function(){
6534                     if(textEl.dom.offsetHeight != textSize){
6535                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6536                     }
6537                 }, this.textResizeInterval);
6538             }
6539             textEvent.addListener(fn, scope, options);
6540         },
6541
6542         /**
6543          * Removes the passed window resize listener.
6544          * @param {Function} fn        The method the event invokes
6545          * @param {Object}   scope    The scope of handler
6546          */
6547         removeResizeListener : function(fn, scope){
6548             if(resizeEvent){
6549                 resizeEvent.removeListener(fn, scope);
6550             }
6551         },
6552
6553         // private
6554         fireResize : function(){
6555             if(resizeEvent){
6556                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6557             }   
6558         },
6559         /**
6560          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6561          */
6562         ieDeferSrc : false,
6563         /**
6564          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6565          */
6566         textResizeInterval : 50
6567     };
6568     
6569     /**
6570      * Fix for doc tools
6571      * @scopeAlias pub=Roo.EventManager
6572      */
6573     
6574      /**
6575      * Appends an event handler to an element (shorthand for addListener)
6576      * @param {String/HTMLElement}   element        The html element or id to assign the
6577      * @param {String}   eventName The type of event to listen for
6578      * @param {Function} handler The method the event invokes
6579      * @param {Object}   scope (optional) The scope in which to execute the handler
6580      * function. The handler function's "this" context.
6581      * @param {Object}   options (optional) An object containing handler configuration
6582      * properties. This may contain any of the following properties:<ul>
6583      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6584      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6585      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6586      * <li>preventDefault {Boolean} True to prevent the default action</li>
6587      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6588      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6589      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6590      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6591      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6592      * by the specified number of milliseconds. If the event fires again within that time, the original
6593      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6594      * </ul><br>
6595      * <p>
6596      * <b>Combining Options</b><br>
6597      * Using the options argument, it is possible to combine different types of listeners:<br>
6598      * <br>
6599      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6600      * Code:<pre><code>
6601 el.on('click', this.onClick, this, {
6602     single: true,
6603     delay: 100,
6604     stopEvent : true,
6605     forumId: 4
6606 });</code></pre>
6607      * <p>
6608      * <b>Attaching multiple handlers in 1 call</b><br>
6609       * The method also allows for a single argument to be passed which is a config object containing properties
6610      * which specify multiple handlers.
6611      * <p>
6612      * Code:<pre><code>
6613 el.on({
6614     'click' : {
6615         fn: this.onClick
6616         scope: this,
6617         delay: 100
6618     },
6619     'mouseover' : {
6620         fn: this.onMouseOver
6621         scope: this
6622     },
6623     'mouseout' : {
6624         fn: this.onMouseOut
6625         scope: this
6626     }
6627 });</code></pre>
6628      * <p>
6629      * Or a shorthand syntax:<br>
6630      * Code:<pre><code>
6631 el.on({
6632     'click' : this.onClick,
6633     'mouseover' : this.onMouseOver,
6634     'mouseout' : this.onMouseOut
6635     scope: this
6636 });</code></pre>
6637      */
6638     pub.on = pub.addListener;
6639     pub.un = pub.removeListener;
6640
6641     pub.stoppedMouseDownEvent = new Roo.util.Event();
6642     return pub;
6643 }();
6644 /**
6645   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6646   * @param {Function} fn        The method the event invokes
6647   * @param {Object}   scope    An  object that becomes the scope of the handler
6648   * @param {boolean}  override If true, the obj passed in becomes
6649   *                             the execution scope of the listener
6650   * @member Roo
6651   * @method onReady
6652  */
6653 Roo.onReady = Roo.EventManager.onDocumentReady;
6654
6655 Roo.onReady(function(){
6656     var bd = Roo.get(document.body);
6657     if(!bd){ return; }
6658
6659     var cls = [
6660             Roo.isIE ? "roo-ie"
6661             : Roo.isIE11 ? "roo-ie11"
6662             : Roo.isEdge ? "roo-edge"
6663             : Roo.isGecko ? "roo-gecko"
6664             : Roo.isOpera ? "roo-opera"
6665             : Roo.isSafari ? "roo-safari" : ""];
6666
6667     if(Roo.isMac){
6668         cls.push("roo-mac");
6669     }
6670     if(Roo.isLinux){
6671         cls.push("roo-linux");
6672     }
6673     if(Roo.isIOS){
6674         cls.push("roo-ios");
6675     }
6676     if(Roo.isTouch){
6677         cls.push("roo-touch");
6678     }
6679     if(Roo.isBorderBox){
6680         cls.push('roo-border-box');
6681     }
6682     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6683         var p = bd.dom.parentNode;
6684         if(p){
6685             p.className += ' roo-strict';
6686         }
6687     }
6688     bd.addClass(cls.join(' '));
6689 });
6690
6691 /**
6692  * @class Roo.EventObject
6693  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6694  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6695  * Example:
6696  * <pre><code>
6697  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6698     e.preventDefault();
6699     var target = e.getTarget();
6700     ...
6701  }
6702  var myDiv = Roo.get("myDiv");
6703  myDiv.on("click", handleClick);
6704  //or
6705  Roo.EventManager.on("myDiv", 'click', handleClick);
6706  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6707  </code></pre>
6708  * @singleton
6709  */
6710 Roo.EventObject = function(){
6711     
6712     var E = Roo.lib.Event;
6713     
6714     // safari keypress events for special keys return bad keycodes
6715     var safariKeys = {
6716         63234 : 37, // left
6717         63235 : 39, // right
6718         63232 : 38, // up
6719         63233 : 40, // down
6720         63276 : 33, // page up
6721         63277 : 34, // page down
6722         63272 : 46, // delete
6723         63273 : 36, // home
6724         63275 : 35  // end
6725     };
6726
6727     // normalize button clicks
6728     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6729                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6730
6731     Roo.EventObjectImpl = function(e){
6732         if(e){
6733             this.setEvent(e.browserEvent || e);
6734         }
6735     };
6736     Roo.EventObjectImpl.prototype = {
6737         /**
6738          * Used to fix doc tools.
6739          * @scope Roo.EventObject.prototype
6740          */
6741             
6742
6743         
6744         
6745         /** The normal browser event */
6746         browserEvent : null,
6747         /** The button pressed in a mouse event */
6748         button : -1,
6749         /** True if the shift key was down during the event */
6750         shiftKey : false,
6751         /** True if the control key was down during the event */
6752         ctrlKey : false,
6753         /** True if the alt key was down during the event */
6754         altKey : false,
6755
6756         /** Key constant 
6757         * @type Number */
6758         BACKSPACE : 8,
6759         /** Key constant 
6760         * @type Number */
6761         TAB : 9,
6762         /** Key constant 
6763         * @type Number */
6764         RETURN : 13,
6765         /** Key constant 
6766         * @type Number */
6767         ENTER : 13,
6768         /** Key constant 
6769         * @type Number */
6770         SHIFT : 16,
6771         /** Key constant 
6772         * @type Number */
6773         CONTROL : 17,
6774         /** Key constant 
6775         * @type Number */
6776         ESC : 27,
6777         /** Key constant 
6778         * @type Number */
6779         SPACE : 32,
6780         /** Key constant 
6781         * @type Number */
6782         PAGEUP : 33,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEDOWN : 34,
6786         /** Key constant 
6787         * @type Number */
6788         END : 35,
6789         /** Key constant 
6790         * @type Number */
6791         HOME : 36,
6792         /** Key constant 
6793         * @type Number */
6794         LEFT : 37,
6795         /** Key constant 
6796         * @type Number */
6797         UP : 38,
6798         /** Key constant 
6799         * @type Number */
6800         RIGHT : 39,
6801         /** Key constant 
6802         * @type Number */
6803         DOWN : 40,
6804         /** Key constant 
6805         * @type Number */
6806         DELETE : 46,
6807         /** Key constant 
6808         * @type Number */
6809         F5 : 116,
6810
6811            /** @private */
6812         setEvent : function(e){
6813             if(e == this || (e && e.browserEvent)){ // already wrapped
6814                 return e;
6815             }
6816             this.browserEvent = e;
6817             if(e){
6818                 // normalize buttons
6819                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6820                 if(e.type == 'click' && this.button == -1){
6821                     this.button = 0;
6822                 }
6823                 this.type = e.type;
6824                 this.shiftKey = e.shiftKey;
6825                 // mac metaKey behaves like ctrlKey
6826                 this.ctrlKey = e.ctrlKey || e.metaKey;
6827                 this.altKey = e.altKey;
6828                 // in getKey these will be normalized for the mac
6829                 this.keyCode = e.keyCode;
6830                 // keyup warnings on firefox.
6831                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6832                 // cache the target for the delayed and or buffered events
6833                 this.target = E.getTarget(e);
6834                 // same for XY
6835                 this.xy = E.getXY(e);
6836             }else{
6837                 this.button = -1;
6838                 this.shiftKey = false;
6839                 this.ctrlKey = false;
6840                 this.altKey = false;
6841                 this.keyCode = 0;
6842                 this.charCode =0;
6843                 this.target = null;
6844                 this.xy = [0, 0];
6845             }
6846             return this;
6847         },
6848
6849         /**
6850          * Stop the event (preventDefault and stopPropagation)
6851          */
6852         stopEvent : function(){
6853             if(this.browserEvent){
6854                 if(this.browserEvent.type == 'mousedown'){
6855                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6856                 }
6857                 E.stopEvent(this.browserEvent);
6858             }
6859         },
6860
6861         /**
6862          * Prevents the browsers default handling of the event.
6863          */
6864         preventDefault : function(){
6865             if(this.browserEvent){
6866                 E.preventDefault(this.browserEvent);
6867             }
6868         },
6869
6870         /** @private */
6871         isNavKeyPress : function(){
6872             var k = this.keyCode;
6873             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6874             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6875         },
6876
6877         isSpecialKey : function(){
6878             var k = this.keyCode;
6879             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6880             (k == 16) || (k == 17) ||
6881             (k >= 18 && k <= 20) ||
6882             (k >= 33 && k <= 35) ||
6883             (k >= 36 && k <= 39) ||
6884             (k >= 44 && k <= 45);
6885         },
6886         /**
6887          * Cancels bubbling of the event.
6888          */
6889         stopPropagation : function(){
6890             if(this.browserEvent){
6891                 if(this.type == 'mousedown'){
6892                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6893                 }
6894                 E.stopPropagation(this.browserEvent);
6895             }
6896         },
6897
6898         /**
6899          * Gets the key code for the event.
6900          * @return {Number}
6901          */
6902         getCharCode : function(){
6903             return this.charCode || this.keyCode;
6904         },
6905
6906         /**
6907          * Returns a normalized keyCode for the event.
6908          * @return {Number} The key code
6909          */
6910         getKey : function(){
6911             var k = this.keyCode || this.charCode;
6912             return Roo.isSafari ? (safariKeys[k] || k) : k;
6913         },
6914
6915         /**
6916          * Gets the x coordinate of the event.
6917          * @return {Number}
6918          */
6919         getPageX : function(){
6920             return this.xy[0];
6921         },
6922
6923         /**
6924          * Gets the y coordinate of the event.
6925          * @return {Number}
6926          */
6927         getPageY : function(){
6928             return this.xy[1];
6929         },
6930
6931         /**
6932          * Gets the time of the event.
6933          * @return {Number}
6934          */
6935         getTime : function(){
6936             if(this.browserEvent){
6937                 return E.getTime(this.browserEvent);
6938             }
6939             return null;
6940         },
6941
6942         /**
6943          * Gets the page coordinates of the event.
6944          * @return {Array} The xy values like [x, y]
6945          */
6946         getXY : function(){
6947             return this.xy;
6948         },
6949
6950         /**
6951          * Gets the target for the event.
6952          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6953          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6954                 search as a number or element (defaults to 10 || document.body)
6955          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6956          * @return {HTMLelement}
6957          */
6958         getTarget : function(selector, maxDepth, returnEl){
6959             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6960         },
6961         /**
6962          * Gets the related target.
6963          * @return {HTMLElement}
6964          */
6965         getRelatedTarget : function(){
6966             if(this.browserEvent){
6967                 return E.getRelatedTarget(this.browserEvent);
6968             }
6969             return null;
6970         },
6971
6972         /**
6973          * Normalizes mouse wheel delta across browsers
6974          * @return {Number} The delta
6975          */
6976         getWheelDelta : function(){
6977             var e = this.browserEvent;
6978             var delta = 0;
6979             if(e.wheelDelta){ /* IE/Opera. */
6980                 delta = e.wheelDelta/120;
6981             }else if(e.detail){ /* Mozilla case. */
6982                 delta = -e.detail/3;
6983             }
6984             return delta;
6985         },
6986
6987         /**
6988          * Returns true if the control, meta, shift or alt key was pressed during this event.
6989          * @return {Boolean}
6990          */
6991         hasModifier : function(){
6992             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6993         },
6994
6995         /**
6996          * Returns true if the target of this event equals el or is a child of el
6997          * @param {String/HTMLElement/Element} el
6998          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
6999          * @return {Boolean}
7000          */
7001         within : function(el, related){
7002             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7003             return t && Roo.fly(el).contains(t);
7004         },
7005
7006         getPoint : function(){
7007             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7008         }
7009     };
7010
7011     return new Roo.EventObjectImpl();
7012 }();
7013             
7014     /*
7015  * Based on:
7016  * Ext JS Library 1.1.1
7017  * Copyright(c) 2006-2007, Ext JS, LLC.
7018  *
7019  * Originally Released Under LGPL - original licence link has changed is not relivant.
7020  *
7021  * Fork - LGPL
7022  * <script type="text/javascript">
7023  */
7024
7025  
7026 // was in Composite Element!??!?!
7027  
7028 (function(){
7029     var D = Roo.lib.Dom;
7030     var E = Roo.lib.Event;
7031     var A = Roo.lib.Anim;
7032
7033     // local style camelizing for speed
7034     var propCache = {};
7035     var camelRe = /(-[a-z])/gi;
7036     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7037     var view = document.defaultView;
7038
7039 /**
7040  * @class Roo.Element
7041  * Represents an Element in the DOM.<br><br>
7042  * Usage:<br>
7043 <pre><code>
7044 var el = Roo.get("my-div");
7045
7046 // or with getEl
7047 var el = getEl("my-div");
7048
7049 // or with a DOM element
7050 var el = Roo.get(myDivElement);
7051 </code></pre>
7052  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7053  * each call instead of constructing a new one.<br><br>
7054  * <b>Animations</b><br />
7055  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7056  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7057 <pre>
7058 Option    Default   Description
7059 --------- --------  ---------------------------------------------
7060 duration  .35       The duration of the animation in seconds
7061 easing    easeOut   The YUI easing method
7062 callback  none      A function to execute when the anim completes
7063 scope     this      The scope (this) of the callback function
7064 </pre>
7065 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7066 * manipulate the animation. Here's an example:
7067 <pre><code>
7068 var el = Roo.get("my-div");
7069
7070 // no animation
7071 el.setWidth(100);
7072
7073 // default animation
7074 el.setWidth(100, true);
7075
7076 // animation with some options set
7077 el.setWidth(100, {
7078     duration: 1,
7079     callback: this.foo,
7080     scope: this
7081 });
7082
7083 // using the "anim" property to get the Anim object
7084 var opt = {
7085     duration: 1,
7086     callback: this.foo,
7087     scope: this
7088 };
7089 el.setWidth(100, opt);
7090 ...
7091 if(opt.anim.isAnimated()){
7092     opt.anim.stop();
7093 }
7094 </code></pre>
7095 * <b> Composite (Collections of) Elements</b><br />
7096  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7097  * @constructor Create a new Element directly.
7098  * @param {String/HTMLElement} element
7099  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7100  */
7101     Roo.Element = function(element, forceNew){
7102         var dom = typeof element == "string" ?
7103                 document.getElementById(element) : element;
7104         if(!dom){ // invalid id/element
7105             return null;
7106         }
7107         var id = dom.id;
7108         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7109             return Roo.Element.cache[id];
7110         }
7111
7112         /**
7113          * The DOM element
7114          * @type HTMLElement
7115          */
7116         this.dom = dom;
7117
7118         /**
7119          * The DOM element ID
7120          * @type String
7121          */
7122         this.id = id || Roo.id(dom);
7123     };
7124
7125     var El = Roo.Element;
7126
7127     El.prototype = {
7128         /**
7129          * The element's default display mode  (defaults to "")
7130          * @type String
7131          */
7132         originalDisplay : "",
7133
7134         visibilityMode : 1,
7135         /**
7136          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7137          * @type String
7138          */
7139         defaultUnit : "px",
7140         
7141         /**
7142          * Sets the element's visibility mode. When setVisible() is called it
7143          * will use this to determine whether to set the visibility or the display property.
7144          * @param visMode Element.VISIBILITY or Element.DISPLAY
7145          * @return {Roo.Element} this
7146          */
7147         setVisibilityMode : function(visMode){
7148             this.visibilityMode = visMode;
7149             return this;
7150         },
7151         /**
7152          * Convenience method for setVisibilityMode(Element.DISPLAY)
7153          * @param {String} display (optional) What to set display to when visible
7154          * @return {Roo.Element} this
7155          */
7156         enableDisplayMode : function(display){
7157             this.setVisibilityMode(El.DISPLAY);
7158             if(typeof display != "undefined") { this.originalDisplay = display; }
7159             return this;
7160         },
7161
7162         /**
7163          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7164          * @param {String} selector The simple selector to test
7165          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7166                 search as a number or element (defaults to 10 || document.body)
7167          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7168          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7169          */
7170         findParent : function(simpleSelector, maxDepth, returnEl){
7171             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7172             maxDepth = maxDepth || 50;
7173             if(typeof maxDepth != "number"){
7174                 stopEl = Roo.getDom(maxDepth);
7175                 maxDepth = 10;
7176             }
7177             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7178                 if(dq.is(p, simpleSelector)){
7179                     return returnEl ? Roo.get(p) : p;
7180                 }
7181                 depth++;
7182                 p = p.parentNode;
7183             }
7184             return null;
7185         },
7186
7187
7188         /**
7189          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7190          * @param {String} selector The simple selector to test
7191          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7192                 search as a number or element (defaults to 10 || document.body)
7193          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7194          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7195          */
7196         findParentNode : function(simpleSelector, maxDepth, returnEl){
7197             var p = Roo.fly(this.dom.parentNode, '_internal');
7198             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7199         },
7200         
7201         /**
7202          * Looks at  the scrollable parent element
7203          */
7204         findScrollableParent : function()
7205         {
7206             var overflowRegex = /(auto|scroll)/;
7207             
7208             if(this.getStyle('position') === 'fixed'){
7209                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7210             }
7211             
7212             var excludeStaticParent = this.getStyle('position') === "absolute";
7213             
7214             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7215                 
7216                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7217                     continue;
7218                 }
7219                 
7220                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7221                     return parent;
7222                 }
7223                 
7224                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7225                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7226                 }
7227             }
7228             
7229             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7230         },
7231
7232         /**
7233          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7234          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7235          * @param {String} selector The simple selector to test
7236          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7237                 search as a number or element (defaults to 10 || document.body)
7238          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7239          */
7240         up : function(simpleSelector, maxDepth){
7241             return this.findParentNode(simpleSelector, maxDepth, true);
7242         },
7243
7244
7245
7246         /**
7247          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7248          * @param {String} selector The simple selector to test
7249          * @return {Boolean} True if this element matches the selector, else false
7250          */
7251         is : function(simpleSelector){
7252             return Roo.DomQuery.is(this.dom, simpleSelector);
7253         },
7254
7255         /**
7256          * Perform animation on this element.
7257          * @param {Object} args The YUI animation control args
7258          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7259          * @param {Function} onComplete (optional) Function to call when animation completes
7260          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7261          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7262          * @return {Roo.Element} this
7263          */
7264         animate : function(args, duration, onComplete, easing, animType){
7265             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7266             return this;
7267         },
7268
7269         /*
7270          * @private Internal animation call
7271          */
7272         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7273             animType = animType || 'run';
7274             opt = opt || {};
7275             var anim = Roo.lib.Anim[animType](
7276                 this.dom, args,
7277                 (opt.duration || defaultDur) || .35,
7278                 (opt.easing || defaultEase) || 'easeOut',
7279                 function(){
7280                     Roo.callback(cb, this);
7281                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7282                 },
7283                 this
7284             );
7285             opt.anim = anim;
7286             return anim;
7287         },
7288
7289         // private legacy anim prep
7290         preanim : function(a, i){
7291             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7292         },
7293
7294         /**
7295          * Removes worthless text nodes
7296          * @param {Boolean} forceReclean (optional) By default the element
7297          * keeps track if it has been cleaned already so
7298          * you can call this over and over. However, if you update the element and
7299          * need to force a reclean, you can pass true.
7300          */
7301         clean : function(forceReclean){
7302             if(this.isCleaned && forceReclean !== true){
7303                 return this;
7304             }
7305             var ns = /\S/;
7306             var d = this.dom, n = d.firstChild, ni = -1;
7307             while(n){
7308                 var nx = n.nextSibling;
7309                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7310                     d.removeChild(n);
7311                 }else{
7312                     n.nodeIndex = ++ni;
7313                 }
7314                 n = nx;
7315             }
7316             this.isCleaned = true;
7317             return this;
7318         },
7319
7320         // private
7321         calcOffsetsTo : function(el){
7322             el = Roo.get(el);
7323             var d = el.dom;
7324             var restorePos = false;
7325             if(el.getStyle('position') == 'static'){
7326                 el.position('relative');
7327                 restorePos = true;
7328             }
7329             var x = 0, y =0;
7330             var op = this.dom;
7331             while(op && op != d && op.tagName != 'HTML'){
7332                 x+= op.offsetLeft;
7333                 y+= op.offsetTop;
7334                 op = op.offsetParent;
7335             }
7336             if(restorePos){
7337                 el.position('static');
7338             }
7339             return [x, y];
7340         },
7341
7342         /**
7343          * Scrolls this element into view within the passed container.
7344          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7345          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7346          * @return {Roo.Element} this
7347          */
7348         scrollIntoView : function(container, hscroll){
7349             var c = Roo.getDom(container) || document.body;
7350             var el = this.dom;
7351
7352             var o = this.calcOffsetsTo(c),
7353                 l = o[0],
7354                 t = o[1],
7355                 b = t+el.offsetHeight,
7356                 r = l+el.offsetWidth;
7357
7358             var ch = c.clientHeight;
7359             var ct = parseInt(c.scrollTop, 10);
7360             var cl = parseInt(c.scrollLeft, 10);
7361             var cb = ct + ch;
7362             var cr = cl + c.clientWidth;
7363
7364             if(t < ct){
7365                 c.scrollTop = t;
7366             }else if(b > cb){
7367                 c.scrollTop = b-ch;
7368             }
7369
7370             if(hscroll !== false){
7371                 if(l < cl){
7372                     c.scrollLeft = l;
7373                 }else if(r > cr){
7374                     c.scrollLeft = r-c.clientWidth;
7375                 }
7376             }
7377             return this;
7378         },
7379
7380         // private
7381         scrollChildIntoView : function(child, hscroll){
7382             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7383         },
7384
7385         /**
7386          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7387          * the new height may not be available immediately.
7388          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7389          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7390          * @param {Function} onComplete (optional) Function to call when animation completes
7391          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7392          * @return {Roo.Element} this
7393          */
7394         autoHeight : function(animate, duration, onComplete, easing){
7395             var oldHeight = this.getHeight();
7396             this.clip();
7397             this.setHeight(1); // force clipping
7398             setTimeout(function(){
7399                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7400                 if(!animate){
7401                     this.setHeight(height);
7402                     this.unclip();
7403                     if(typeof onComplete == "function"){
7404                         onComplete();
7405                     }
7406                 }else{
7407                     this.setHeight(oldHeight); // restore original height
7408                     this.setHeight(height, animate, duration, function(){
7409                         this.unclip();
7410                         if(typeof onComplete == "function") { onComplete(); }
7411                     }.createDelegate(this), easing);
7412                 }
7413             }.createDelegate(this), 0);
7414             return this;
7415         },
7416
7417         /**
7418          * Returns true if this element is an ancestor of the passed element
7419          * @param {HTMLElement/String} el The element to check
7420          * @return {Boolean} True if this element is an ancestor of el, else false
7421          */
7422         contains : function(el){
7423             if(!el){return false;}
7424             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7425         },
7426
7427         /**
7428          * Checks whether the element is currently visible using both visibility and display properties.
7429          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7430          * @return {Boolean} True if the element is currently visible, else false
7431          */
7432         isVisible : function(deep) {
7433             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7434             if(deep !== true || !vis){
7435                 return vis;
7436             }
7437             var p = this.dom.parentNode;
7438             while(p && p.tagName.toLowerCase() != "body"){
7439                 if(!Roo.fly(p, '_isVisible').isVisible()){
7440                     return false;
7441                 }
7442                 p = p.parentNode;
7443             }
7444             return true;
7445         },
7446
7447         /**
7448          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7449          * @param {String} selector The CSS selector
7450          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7451          * @return {CompositeElement/CompositeElementLite} The composite element
7452          */
7453         select : function(selector, unique){
7454             return El.select(selector, unique, this.dom);
7455         },
7456
7457         /**
7458          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7459          * @param {String} selector The CSS selector
7460          * @return {Array} An array of the matched nodes
7461          */
7462         query : function(selector, unique){
7463             return Roo.DomQuery.select(selector, this.dom);
7464         },
7465
7466         /**
7467          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7468          * @param {String} selector The CSS selector
7469          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7470          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7471          */
7472         child : function(selector, returnDom){
7473             var n = Roo.DomQuery.selectNode(selector, this.dom);
7474             return returnDom ? n : Roo.get(n);
7475         },
7476
7477         /**
7478          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7479          * @param {String} selector The CSS selector
7480          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7481          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7482          */
7483         down : function(selector, returnDom){
7484             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7485             return returnDom ? n : Roo.get(n);
7486         },
7487
7488         /**
7489          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7490          * @param {String} group The group the DD object is member of
7491          * @param {Object} config The DD config object
7492          * @param {Object} overrides An object containing methods to override/implement on the DD object
7493          * @return {Roo.dd.DD} The DD object
7494          */
7495         initDD : function(group, config, overrides){
7496             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7497             return Roo.apply(dd, overrides);
7498         },
7499
7500         /**
7501          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7502          * @param {String} group The group the DDProxy object is member of
7503          * @param {Object} config The DDProxy config object
7504          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7505          * @return {Roo.dd.DDProxy} The DDProxy object
7506          */
7507         initDDProxy : function(group, config, overrides){
7508             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7509             return Roo.apply(dd, overrides);
7510         },
7511
7512         /**
7513          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7514          * @param {String} group The group the DDTarget object is member of
7515          * @param {Object} config The DDTarget config object
7516          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7517          * @return {Roo.dd.DDTarget} The DDTarget object
7518          */
7519         initDDTarget : function(group, config, overrides){
7520             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7521             return Roo.apply(dd, overrides);
7522         },
7523
7524         /**
7525          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7526          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7527          * @param {Boolean} visible Whether the element is visible
7528          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7529          * @return {Roo.Element} this
7530          */
7531          setVisible : function(visible, animate){
7532             if(!animate || !A){
7533                 if(this.visibilityMode == El.DISPLAY){
7534                     this.setDisplayed(visible);
7535                 }else{
7536                     this.fixDisplay();
7537                     this.dom.style.visibility = visible ? "visible" : "hidden";
7538                 }
7539             }else{
7540                 // closure for composites
7541                 var dom = this.dom;
7542                 var visMode = this.visibilityMode;
7543                 if(visible){
7544                     this.setOpacity(.01);
7545                     this.setVisible(true);
7546                 }
7547                 this.anim({opacity: { to: (visible?1:0) }},
7548                       this.preanim(arguments, 1),
7549                       null, .35, 'easeIn', function(){
7550                          if(!visible){
7551                              if(visMode == El.DISPLAY){
7552                                  dom.style.display = "none";
7553                              }else{
7554                                  dom.style.visibility = "hidden";
7555                              }
7556                              Roo.get(dom).setOpacity(1);
7557                          }
7558                      });
7559             }
7560             return this;
7561         },
7562
7563         /**
7564          * Returns true if display is not "none"
7565          * @return {Boolean}
7566          */
7567         isDisplayed : function() {
7568             return this.getStyle("display") != "none";
7569         },
7570
7571         /**
7572          * Toggles the element's visibility or display, depending on visibility mode.
7573          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7574          * @return {Roo.Element} this
7575          */
7576         toggle : function(animate){
7577             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7578             return this;
7579         },
7580
7581         /**
7582          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7583          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7584          * @return {Roo.Element} this
7585          */
7586         setDisplayed : function(value) {
7587             if(typeof value == "boolean"){
7588                value = value ? this.originalDisplay : "none";
7589             }
7590             this.setStyle("display", value);
7591             return this;
7592         },
7593
7594         /**
7595          * Tries to focus the element. Any exceptions are caught and ignored.
7596          * @return {Roo.Element} this
7597          */
7598         focus : function() {
7599             try{
7600                 this.dom.focus();
7601             }catch(e){}
7602             return this;
7603         },
7604
7605         /**
7606          * Tries to blur the element. Any exceptions are caught and ignored.
7607          * @return {Roo.Element} this
7608          */
7609         blur : function() {
7610             try{
7611                 this.dom.blur();
7612             }catch(e){}
7613             return this;
7614         },
7615
7616         /**
7617          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7618          * @param {String/Array} className The CSS class to add, or an array of classes
7619          * @return {Roo.Element} this
7620          */
7621         addClass : function(className){
7622             if(className instanceof Array){
7623                 for(var i = 0, len = className.length; i < len; i++) {
7624                     this.addClass(className[i]);
7625                 }
7626             }else{
7627                 if(className && !this.hasClass(className)){
7628                     this.dom.className = this.dom.className + " " + className;
7629                 }
7630             }
7631             return this;
7632         },
7633
7634         /**
7635          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7636          * @param {String/Array} className The CSS class to add, or an array of classes
7637          * @return {Roo.Element} this
7638          */
7639         radioClass : function(className){
7640             var siblings = this.dom.parentNode.childNodes;
7641             for(var i = 0; i < siblings.length; i++) {
7642                 var s = siblings[i];
7643                 if(s.nodeType == 1){
7644                     Roo.get(s).removeClass(className);
7645                 }
7646             }
7647             this.addClass(className);
7648             return this;
7649         },
7650
7651         /**
7652          * Removes one or more CSS classes from the element.
7653          * @param {String/Array} className The CSS class to remove, or an array of classes
7654          * @return {Roo.Element} this
7655          */
7656         removeClass : function(className){
7657             if(!className || !this.dom.className){
7658                 return this;
7659             }
7660             if(className instanceof Array){
7661                 for(var i = 0, len = className.length; i < len; i++) {
7662                     this.removeClass(className[i]);
7663                 }
7664             }else{
7665                 if(this.hasClass(className)){
7666                     var re = this.classReCache[className];
7667                     if (!re) {
7668                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7669                        this.classReCache[className] = re;
7670                     }
7671                     this.dom.className =
7672                         this.dom.className.replace(re, " ");
7673                 }
7674             }
7675             return this;
7676         },
7677
7678         // private
7679         classReCache: {},
7680
7681         /**
7682          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7683          * @param {String} className The CSS class to toggle
7684          * @return {Roo.Element} this
7685          */
7686         toggleClass : function(className){
7687             if(this.hasClass(className)){
7688                 this.removeClass(className);
7689             }else{
7690                 this.addClass(className);
7691             }
7692             return this;
7693         },
7694
7695         /**
7696          * Checks if the specified CSS class exists on this element's DOM node.
7697          * @param {String} className The CSS class to check for
7698          * @return {Boolean} True if the class exists, else false
7699          */
7700         hasClass : function(className){
7701             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7702         },
7703
7704         /**
7705          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7706          * @param {String} oldClassName The CSS class to replace
7707          * @param {String} newClassName The replacement CSS class
7708          * @return {Roo.Element} this
7709          */
7710         replaceClass : function(oldClassName, newClassName){
7711             this.removeClass(oldClassName);
7712             this.addClass(newClassName);
7713             return this;
7714         },
7715
7716         /**
7717          * Returns an object with properties matching the styles requested.
7718          * For example, el.getStyles('color', 'font-size', 'width') might return
7719          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7720          * @param {String} style1 A style name
7721          * @param {String} style2 A style name
7722          * @param {String} etc.
7723          * @return {Object} The style object
7724          */
7725         getStyles : function(){
7726             var a = arguments, len = a.length, r = {};
7727             for(var i = 0; i < len; i++){
7728                 r[a[i]] = this.getStyle(a[i]);
7729             }
7730             return r;
7731         },
7732
7733         /**
7734          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7735          * @param {String} property The style property whose value is returned.
7736          * @return {String} The current value of the style property for this element.
7737          */
7738         getStyle : function(){
7739             return view && view.getComputedStyle ?
7740                 function(prop){
7741                     var el = this.dom, v, cs, camel;
7742                     if(prop == 'float'){
7743                         prop = "cssFloat";
7744                     }
7745                     if(el.style && (v = el.style[prop])){
7746                         return v;
7747                     }
7748                     if(cs = view.getComputedStyle(el, "")){
7749                         if(!(camel = propCache[prop])){
7750                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7751                         }
7752                         return cs[camel];
7753                     }
7754                     return null;
7755                 } :
7756                 function(prop){
7757                     var el = this.dom, v, cs, camel;
7758                     if(prop == 'opacity'){
7759                         if(typeof el.style.filter == 'string'){
7760                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7761                             if(m){
7762                                 var fv = parseFloat(m[1]);
7763                                 if(!isNaN(fv)){
7764                                     return fv ? fv / 100 : 0;
7765                                 }
7766                             }
7767                         }
7768                         return 1;
7769                     }else if(prop == 'float'){
7770                         prop = "styleFloat";
7771                     }
7772                     if(!(camel = propCache[prop])){
7773                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7774                     }
7775                     if(v = el.style[camel]){
7776                         return v;
7777                     }
7778                     if(cs = el.currentStyle){
7779                         return cs[camel];
7780                     }
7781                     return null;
7782                 };
7783         }(),
7784
7785         /**
7786          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7787          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7788          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7789          * @return {Roo.Element} this
7790          */
7791         setStyle : function(prop, value){
7792             if(typeof prop == "string"){
7793                 
7794                 if (prop == 'float') {
7795                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7796                     return this;
7797                 }
7798                 
7799                 var camel;
7800                 if(!(camel = propCache[prop])){
7801                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7802                 }
7803                 
7804                 if(camel == 'opacity') {
7805                     this.setOpacity(value);
7806                 }else{
7807                     this.dom.style[camel] = value;
7808                 }
7809             }else{
7810                 for(var style in prop){
7811                     if(typeof prop[style] != "function"){
7812                        this.setStyle(style, prop[style]);
7813                     }
7814                 }
7815             }
7816             return this;
7817         },
7818
7819         /**
7820          * More flexible version of {@link #setStyle} for setting style properties.
7821          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7822          * a function which returns such a specification.
7823          * @return {Roo.Element} this
7824          */
7825         applyStyles : function(style){
7826             Roo.DomHelper.applyStyles(this.dom, style);
7827             return this;
7828         },
7829
7830         /**
7831           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7832           * @return {Number} The X position of the element
7833           */
7834         getX : function(){
7835             return D.getX(this.dom);
7836         },
7837
7838         /**
7839           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7840           * @return {Number} The Y position of the element
7841           */
7842         getY : function(){
7843             return D.getY(this.dom);
7844         },
7845
7846         /**
7847           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7848           * @return {Array} The XY position of the element
7849           */
7850         getXY : function(){
7851             return D.getXY(this.dom);
7852         },
7853
7854         /**
7855          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7856          * @param {Number} The X position of the element
7857          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7858          * @return {Roo.Element} this
7859          */
7860         setX : function(x, animate){
7861             if(!animate || !A){
7862                 D.setX(this.dom, x);
7863             }else{
7864                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7865             }
7866             return this;
7867         },
7868
7869         /**
7870          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7871          * @param {Number} The Y position of the element
7872          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7873          * @return {Roo.Element} this
7874          */
7875         setY : function(y, animate){
7876             if(!animate || !A){
7877                 D.setY(this.dom, y);
7878             }else{
7879                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7880             }
7881             return this;
7882         },
7883
7884         /**
7885          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7886          * @param {String} left The left CSS property value
7887          * @return {Roo.Element} this
7888          */
7889         setLeft : function(left){
7890             this.setStyle("left", this.addUnits(left));
7891             return this;
7892         },
7893
7894         /**
7895          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7896          * @param {String} top The top CSS property value
7897          * @return {Roo.Element} this
7898          */
7899         setTop : function(top){
7900             this.setStyle("top", this.addUnits(top));
7901             return this;
7902         },
7903
7904         /**
7905          * Sets the element's CSS right style.
7906          * @param {String} right The right CSS property value
7907          * @return {Roo.Element} this
7908          */
7909         setRight : function(right){
7910             this.setStyle("right", this.addUnits(right));
7911             return this;
7912         },
7913
7914         /**
7915          * Sets the element's CSS bottom style.
7916          * @param {String} bottom The bottom CSS property value
7917          * @return {Roo.Element} this
7918          */
7919         setBottom : function(bottom){
7920             this.setStyle("bottom", this.addUnits(bottom));
7921             return this;
7922         },
7923
7924         /**
7925          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7926          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7927          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7928          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7929          * @return {Roo.Element} this
7930          */
7931         setXY : function(pos, animate){
7932             if(!animate || !A){
7933                 D.setXY(this.dom, pos);
7934             }else{
7935                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7936             }
7937             return this;
7938         },
7939
7940         /**
7941          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7942          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7943          * @param {Number} x X value for new position (coordinates are page-based)
7944          * @param {Number} y Y value for new position (coordinates are page-based)
7945          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7946          * @return {Roo.Element} this
7947          */
7948         setLocation : function(x, y, animate){
7949             this.setXY([x, y], this.preanim(arguments, 2));
7950             return this;
7951         },
7952
7953         /**
7954          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7955          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7956          * @param {Number} x X value for new position (coordinates are page-based)
7957          * @param {Number} y Y value for new position (coordinates are page-based)
7958          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7959          * @return {Roo.Element} this
7960          */
7961         moveTo : function(x, y, animate){
7962             this.setXY([x, y], this.preanim(arguments, 2));
7963             return this;
7964         },
7965
7966         /**
7967          * Returns the region of the given element.
7968          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7969          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7970          */
7971         getRegion : function(){
7972             return D.getRegion(this.dom);
7973         },
7974
7975         /**
7976          * Returns the offset height of the element
7977          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7978          * @return {Number} The element's height
7979          */
7980         getHeight : function(contentHeight){
7981             var h = this.dom.offsetHeight || 0;
7982             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7983         },
7984
7985         /**
7986          * Returns the offset width of the element
7987          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7988          * @return {Number} The element's width
7989          */
7990         getWidth : function(contentWidth){
7991             var w = this.dom.offsetWidth || 0;
7992             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7993         },
7994
7995         /**
7996          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
7997          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
7998          * if a height has not been set using CSS.
7999          * @return {Number}
8000          */
8001         getComputedHeight : function(){
8002             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8003             if(!h){
8004                 h = parseInt(this.getStyle('height'), 10) || 0;
8005                 if(!this.isBorderBox()){
8006                     h += this.getFrameWidth('tb');
8007                 }
8008             }
8009             return h;
8010         },
8011
8012         /**
8013          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8014          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8015          * if a width has not been set using CSS.
8016          * @return {Number}
8017          */
8018         getComputedWidth : function(){
8019             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8020             if(!w){
8021                 w = parseInt(this.getStyle('width'), 10) || 0;
8022                 if(!this.isBorderBox()){
8023                     w += this.getFrameWidth('lr');
8024                 }
8025             }
8026             return w;
8027         },
8028
8029         /**
8030          * Returns the size of the element.
8031          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8032          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8033          */
8034         getSize : function(contentSize){
8035             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8036         },
8037
8038         /**
8039          * Returns the width and height of the viewport.
8040          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8041          */
8042         getViewSize : function(){
8043             var d = this.dom, doc = document, aw = 0, ah = 0;
8044             if(d == doc || d == doc.body){
8045                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8046             }else{
8047                 return {
8048                     width : d.clientWidth,
8049                     height: d.clientHeight
8050                 };
8051             }
8052         },
8053
8054         /**
8055          * Returns the value of the "value" attribute
8056          * @param {Boolean} asNumber true to parse the value as a number
8057          * @return {String/Number}
8058          */
8059         getValue : function(asNumber){
8060             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8061         },
8062
8063         // private
8064         adjustWidth : function(width){
8065             if(typeof width == "number"){
8066                 if(this.autoBoxAdjust && !this.isBorderBox()){
8067                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8068                 }
8069                 if(width < 0){
8070                     width = 0;
8071                 }
8072             }
8073             return width;
8074         },
8075
8076         // private
8077         adjustHeight : function(height){
8078             if(typeof height == "number"){
8079                if(this.autoBoxAdjust && !this.isBorderBox()){
8080                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8081                }
8082                if(height < 0){
8083                    height = 0;
8084                }
8085             }
8086             return height;
8087         },
8088
8089         /**
8090          * Set the width of the element
8091          * @param {Number} width The new width
8092          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8093          * @return {Roo.Element} this
8094          */
8095         setWidth : function(width, animate){
8096             width = this.adjustWidth(width);
8097             if(!animate || !A){
8098                 this.dom.style.width = this.addUnits(width);
8099             }else{
8100                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8101             }
8102             return this;
8103         },
8104
8105         /**
8106          * Set the height of the element
8107          * @param {Number} height The new height
8108          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8109          * @return {Roo.Element} this
8110          */
8111          setHeight : function(height, animate){
8112             height = this.adjustHeight(height);
8113             if(!animate || !A){
8114                 this.dom.style.height = this.addUnits(height);
8115             }else{
8116                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8117             }
8118             return this;
8119         },
8120
8121         /**
8122          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8123          * @param {Number} width The new width
8124          * @param {Number} height The new height
8125          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8126          * @return {Roo.Element} this
8127          */
8128          setSize : function(width, height, animate){
8129             if(typeof width == "object"){ // in case of object from getSize()
8130                 height = width.height; width = width.width;
8131             }
8132             width = this.adjustWidth(width); height = this.adjustHeight(height);
8133             if(!animate || !A){
8134                 this.dom.style.width = this.addUnits(width);
8135                 this.dom.style.height = this.addUnits(height);
8136             }else{
8137                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8138             }
8139             return this;
8140         },
8141
8142         /**
8143          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8144          * @param {Number} x X value for new position (coordinates are page-based)
8145          * @param {Number} y Y value for new position (coordinates are page-based)
8146          * @param {Number} width The new width
8147          * @param {Number} height The new height
8148          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8149          * @return {Roo.Element} this
8150          */
8151         setBounds : function(x, y, width, height, animate){
8152             if(!animate || !A){
8153                 this.setSize(width, height);
8154                 this.setLocation(x, y);
8155             }else{
8156                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8157                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8158                               this.preanim(arguments, 4), 'motion');
8159             }
8160             return this;
8161         },
8162
8163         /**
8164          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8165          * @param {Roo.lib.Region} region The region to fill
8166          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8167          * @return {Roo.Element} this
8168          */
8169         setRegion : function(region, animate){
8170             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8171             return this;
8172         },
8173
8174         /**
8175          * Appends an event handler
8176          *
8177          * @param {String}   eventName     The type of event to append
8178          * @param {Function} fn        The method the event invokes
8179          * @param {Object} scope       (optional) The scope (this object) of the fn
8180          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8181          */
8182         addListener : function(eventName, fn, scope, options){
8183             if (this.dom) {
8184                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8185             }
8186         },
8187
8188         /**
8189          * Removes an event handler from this element
8190          * @param {String} eventName the type of event to remove
8191          * @param {Function} fn the method the event invokes
8192          * @return {Roo.Element} this
8193          */
8194         removeListener : function(eventName, fn){
8195             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8196             return this;
8197         },
8198
8199         /**
8200          * Removes all previous added listeners from this element
8201          * @return {Roo.Element} this
8202          */
8203         removeAllListeners : function(){
8204             E.purgeElement(this.dom);
8205             return this;
8206         },
8207
8208         relayEvent : function(eventName, observable){
8209             this.on(eventName, function(e){
8210                 observable.fireEvent(eventName, e);
8211             });
8212         },
8213
8214         /**
8215          * Set the opacity of the element
8216          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8217          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8218          * @return {Roo.Element} this
8219          */
8220          setOpacity : function(opacity, animate){
8221             if(!animate || !A){
8222                 var s = this.dom.style;
8223                 if(Roo.isIE){
8224                     s.zoom = 1;
8225                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8226                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8227                 }else{
8228                     s.opacity = opacity;
8229                 }
8230             }else{
8231                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8232             }
8233             return this;
8234         },
8235
8236         /**
8237          * Gets the left X coordinate
8238          * @param {Boolean} local True to get the local css position instead of page coordinate
8239          * @return {Number}
8240          */
8241         getLeft : function(local){
8242             if(!local){
8243                 return this.getX();
8244             }else{
8245                 return parseInt(this.getStyle("left"), 10) || 0;
8246             }
8247         },
8248
8249         /**
8250          * Gets the right X coordinate of the element (element X position + element width)
8251          * @param {Boolean} local True to get the local css position instead of page coordinate
8252          * @return {Number}
8253          */
8254         getRight : function(local){
8255             if(!local){
8256                 return this.getX() + this.getWidth();
8257             }else{
8258                 return (this.getLeft(true) + this.getWidth()) || 0;
8259             }
8260         },
8261
8262         /**
8263          * Gets the top Y coordinate
8264          * @param {Boolean} local True to get the local css position instead of page coordinate
8265          * @return {Number}
8266          */
8267         getTop : function(local) {
8268             if(!local){
8269                 return this.getY();
8270             }else{
8271                 return parseInt(this.getStyle("top"), 10) || 0;
8272             }
8273         },
8274
8275         /**
8276          * Gets the bottom Y coordinate of the element (element Y position + element height)
8277          * @param {Boolean} local True to get the local css position instead of page coordinate
8278          * @return {Number}
8279          */
8280         getBottom : function(local){
8281             if(!local){
8282                 return this.getY() + this.getHeight();
8283             }else{
8284                 return (this.getTop(true) + this.getHeight()) || 0;
8285             }
8286         },
8287
8288         /**
8289         * Initializes positioning on this element. If a desired position is not passed, it will make the
8290         * the element positioned relative IF it is not already positioned.
8291         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8292         * @param {Number} zIndex (optional) The zIndex to apply
8293         * @param {Number} x (optional) Set the page X position
8294         * @param {Number} y (optional) Set the page Y position
8295         */
8296         position : function(pos, zIndex, x, y){
8297             if(!pos){
8298                if(this.getStyle('position') == 'static'){
8299                    this.setStyle('position', 'relative');
8300                }
8301             }else{
8302                 this.setStyle("position", pos);
8303             }
8304             if(zIndex){
8305                 this.setStyle("z-index", zIndex);
8306             }
8307             if(x !== undefined && y !== undefined){
8308                 this.setXY([x, y]);
8309             }else if(x !== undefined){
8310                 this.setX(x);
8311             }else if(y !== undefined){
8312                 this.setY(y);
8313             }
8314         },
8315
8316         /**
8317         * Clear positioning back to the default when the document was loaded
8318         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8319         * @return {Roo.Element} this
8320          */
8321         clearPositioning : function(value){
8322             value = value ||'';
8323             this.setStyle({
8324                 "left": value,
8325                 "right": value,
8326                 "top": value,
8327                 "bottom": value,
8328                 "z-index": "",
8329                 "position" : "static"
8330             });
8331             return this;
8332         },
8333
8334         /**
8335         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8336         * snapshot before performing an update and then restoring the element.
8337         * @return {Object}
8338         */
8339         getPositioning : function(){
8340             var l = this.getStyle("left");
8341             var t = this.getStyle("top");
8342             return {
8343                 "position" : this.getStyle("position"),
8344                 "left" : l,
8345                 "right" : l ? "" : this.getStyle("right"),
8346                 "top" : t,
8347                 "bottom" : t ? "" : this.getStyle("bottom"),
8348                 "z-index" : this.getStyle("z-index")
8349             };
8350         },
8351
8352         /**
8353          * Gets the width of the border(s) for the specified side(s)
8354          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8355          * passing lr would get the border (l)eft width + the border (r)ight width.
8356          * @return {Number} The width of the sides passed added together
8357          */
8358         getBorderWidth : function(side){
8359             return this.addStyles(side, El.borders);
8360         },
8361
8362         /**
8363          * Gets the width of the padding(s) for the specified side(s)
8364          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8365          * passing lr would get the padding (l)eft + the padding (r)ight.
8366          * @return {Number} The padding of the sides passed added together
8367          */
8368         getPadding : function(side){
8369             return this.addStyles(side, El.paddings);
8370         },
8371
8372         /**
8373         * Set positioning with an object returned by getPositioning().
8374         * @param {Object} posCfg
8375         * @return {Roo.Element} this
8376          */
8377         setPositioning : function(pc){
8378             this.applyStyles(pc);
8379             if(pc.right == "auto"){
8380                 this.dom.style.right = "";
8381             }
8382             if(pc.bottom == "auto"){
8383                 this.dom.style.bottom = "";
8384             }
8385             return this;
8386         },
8387
8388         // private
8389         fixDisplay : function(){
8390             if(this.getStyle("display") == "none"){
8391                 this.setStyle("visibility", "hidden");
8392                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8393                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8394                     this.setStyle("display", "block");
8395                 }
8396             }
8397         },
8398
8399         /**
8400          * Quick set left and top adding default units
8401          * @param {String} left The left CSS property value
8402          * @param {String} top The top CSS property value
8403          * @return {Roo.Element} this
8404          */
8405          setLeftTop : function(left, top){
8406             this.dom.style.left = this.addUnits(left);
8407             this.dom.style.top = this.addUnits(top);
8408             return this;
8409         },
8410
8411         /**
8412          * Move this element relative to its current position.
8413          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8414          * @param {Number} distance How far to move the element in pixels
8415          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8416          * @return {Roo.Element} this
8417          */
8418          move : function(direction, distance, animate){
8419             var xy = this.getXY();
8420             direction = direction.toLowerCase();
8421             switch(direction){
8422                 case "l":
8423                 case "left":
8424                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8425                     break;
8426                case "r":
8427                case "right":
8428                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8429                     break;
8430                case "t":
8431                case "top":
8432                case "up":
8433                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8434                     break;
8435                case "b":
8436                case "bottom":
8437                case "down":
8438                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8439                     break;
8440             }
8441             return this;
8442         },
8443
8444         /**
8445          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8446          * @return {Roo.Element} this
8447          */
8448         clip : function(){
8449             if(!this.isClipped){
8450                this.isClipped = true;
8451                this.originalClip = {
8452                    "o": this.getStyle("overflow"),
8453                    "x": this.getStyle("overflow-x"),
8454                    "y": this.getStyle("overflow-y")
8455                };
8456                this.setStyle("overflow", "hidden");
8457                this.setStyle("overflow-x", "hidden");
8458                this.setStyle("overflow-y", "hidden");
8459             }
8460             return this;
8461         },
8462
8463         /**
8464          *  Return clipping (overflow) to original clipping before clip() was called
8465          * @return {Roo.Element} this
8466          */
8467         unclip : function(){
8468             if(this.isClipped){
8469                 this.isClipped = false;
8470                 var o = this.originalClip;
8471                 if(o.o){this.setStyle("overflow", o.o);}
8472                 if(o.x){this.setStyle("overflow-x", o.x);}
8473                 if(o.y){this.setStyle("overflow-y", o.y);}
8474             }
8475             return this;
8476         },
8477
8478
8479         /**
8480          * Gets the x,y coordinates specified by the anchor position on the element.
8481          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8482          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8483          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8484          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8485          * @return {Array} [x, y] An array containing the element's x and y coordinates
8486          */
8487         getAnchorXY : function(anchor, local, s){
8488             //Passing a different size is useful for pre-calculating anchors,
8489             //especially for anchored animations that change the el size.
8490
8491             var w, h, vp = false;
8492             if(!s){
8493                 var d = this.dom;
8494                 if(d == document.body || d == document){
8495                     vp = true;
8496                     w = D.getViewWidth(); h = D.getViewHeight();
8497                 }else{
8498                     w = this.getWidth(); h = this.getHeight();
8499                 }
8500             }else{
8501                 w = s.width;  h = s.height;
8502             }
8503             var x = 0, y = 0, r = Math.round;
8504             switch((anchor || "tl").toLowerCase()){
8505                 case "c":
8506                     x = r(w*.5);
8507                     y = r(h*.5);
8508                 break;
8509                 case "t":
8510                     x = r(w*.5);
8511                     y = 0;
8512                 break;
8513                 case "l":
8514                     x = 0;
8515                     y = r(h*.5);
8516                 break;
8517                 case "r":
8518                     x = w;
8519                     y = r(h*.5);
8520                 break;
8521                 case "b":
8522                     x = r(w*.5);
8523                     y = h;
8524                 break;
8525                 case "tl":
8526                     x = 0;
8527                     y = 0;
8528                 break;
8529                 case "bl":
8530                     x = 0;
8531                     y = h;
8532                 break;
8533                 case "br":
8534                     x = w;
8535                     y = h;
8536                 break;
8537                 case "tr":
8538                     x = w;
8539                     y = 0;
8540                 break;
8541             }
8542             if(local === true){
8543                 return [x, y];
8544             }
8545             if(vp){
8546                 var sc = this.getScroll();
8547                 return [x + sc.left, y + sc.top];
8548             }
8549             //Add the element's offset xy
8550             var o = this.getXY();
8551             return [x+o[0], y+o[1]];
8552         },
8553
8554         /**
8555          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8556          * supported position values.
8557          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8558          * @param {String} position The position to align to.
8559          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8560          * @return {Array} [x, y]
8561          */
8562         getAlignToXY : function(el, p, o){
8563             el = Roo.get(el);
8564             var d = this.dom;
8565             if(!el.dom){
8566                 throw "Element.alignTo with an element that doesn't exist";
8567             }
8568             var c = false; //constrain to viewport
8569             var p1 = "", p2 = "";
8570             o = o || [0,0];
8571
8572             if(!p){
8573                 p = "tl-bl";
8574             }else if(p == "?"){
8575                 p = "tl-bl?";
8576             }else if(p.indexOf("-") == -1){
8577                 p = "tl-" + p;
8578             }
8579             p = p.toLowerCase();
8580             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8581             if(!m){
8582                throw "Element.alignTo with an invalid alignment " + p;
8583             }
8584             p1 = m[1]; p2 = m[2]; c = !!m[3];
8585
8586             //Subtract the aligned el's internal xy from the target's offset xy
8587             //plus custom offset to get the aligned el's new offset xy
8588             var a1 = this.getAnchorXY(p1, true);
8589             var a2 = el.getAnchorXY(p2, false);
8590             var x = a2[0] - a1[0] + o[0];
8591             var y = a2[1] - a1[1] + o[1];
8592             if(c){
8593                 //constrain the aligned el to viewport if necessary
8594                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8595                 // 5px of margin for ie
8596                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8597
8598                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8599                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8600                 //otherwise swap the aligned el to the opposite border of the target.
8601                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8602                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8603                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8604                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8605
8606                var doc = document;
8607                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8608                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8609
8610                if((x+w) > dw + scrollX){
8611                     x = swapX ? r.left-w : dw+scrollX-w;
8612                 }
8613                if(x < scrollX){
8614                    x = swapX ? r.right : scrollX;
8615                }
8616                if((y+h) > dh + scrollY){
8617                     y = swapY ? r.top-h : dh+scrollY-h;
8618                 }
8619                if (y < scrollY){
8620                    y = swapY ? r.bottom : scrollY;
8621                }
8622             }
8623             return [x,y];
8624         },
8625
8626         // private
8627         getConstrainToXY : function(){
8628             var os = {top:0, left:0, bottom:0, right: 0};
8629
8630             return function(el, local, offsets, proposedXY){
8631                 el = Roo.get(el);
8632                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8633
8634                 var vw, vh, vx = 0, vy = 0;
8635                 if(el.dom == document.body || el.dom == document){
8636                     vw = Roo.lib.Dom.getViewWidth();
8637                     vh = Roo.lib.Dom.getViewHeight();
8638                 }else{
8639                     vw = el.dom.clientWidth;
8640                     vh = el.dom.clientHeight;
8641                     if(!local){
8642                         var vxy = el.getXY();
8643                         vx = vxy[0];
8644                         vy = vxy[1];
8645                     }
8646                 }
8647
8648                 var s = el.getScroll();
8649
8650                 vx += offsets.left + s.left;
8651                 vy += offsets.top + s.top;
8652
8653                 vw -= offsets.right;
8654                 vh -= offsets.bottom;
8655
8656                 var vr = vx+vw;
8657                 var vb = vy+vh;
8658
8659                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8660                 var x = xy[0], y = xy[1];
8661                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8662
8663                 // only move it if it needs it
8664                 var moved = false;
8665
8666                 // first validate right/bottom
8667                 if((x + w) > vr){
8668                     x = vr - w;
8669                     moved = true;
8670                 }
8671                 if((y + h) > vb){
8672                     y = vb - h;
8673                     moved = true;
8674                 }
8675                 // then make sure top/left isn't negative
8676                 if(x < vx){
8677                     x = vx;
8678                     moved = true;
8679                 }
8680                 if(y < vy){
8681                     y = vy;
8682                     moved = true;
8683                 }
8684                 return moved ? [x, y] : false;
8685             };
8686         }(),
8687
8688         // private
8689         adjustForConstraints : function(xy, parent, offsets){
8690             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8691         },
8692
8693         /**
8694          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8695          * document it aligns it to the viewport.
8696          * The position parameter is optional, and can be specified in any one of the following formats:
8697          * <ul>
8698          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8699          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8700          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8701          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8702          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8703          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8704          * </ul>
8705          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8706          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8707          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8708          * that specified in order to enforce the viewport constraints.
8709          * Following are all of the supported anchor positions:
8710     <pre>
8711     Value  Description
8712     -----  -----------------------------
8713     tl     The top left corner (default)
8714     t      The center of the top edge
8715     tr     The top right corner
8716     l      The center of the left edge
8717     c      In the center of the element
8718     r      The center of the right edge
8719     bl     The bottom left corner
8720     b      The center of the bottom edge
8721     br     The bottom right corner
8722     </pre>
8723     Example Usage:
8724     <pre><code>
8725     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8726     el.alignTo("other-el");
8727
8728     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8729     el.alignTo("other-el", "tr?");
8730
8731     // align the bottom right corner of el with the center left edge of other-el
8732     el.alignTo("other-el", "br-l?");
8733
8734     // align the center of el with the bottom left corner of other-el and
8735     // adjust the x position by -6 pixels (and the y position by 0)
8736     el.alignTo("other-el", "c-bl", [-6, 0]);
8737     </code></pre>
8738          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8739          * @param {String} position The position to align to.
8740          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8741          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8742          * @return {Roo.Element} this
8743          */
8744         alignTo : function(element, position, offsets, animate){
8745             var xy = this.getAlignToXY(element, position, offsets);
8746             this.setXY(xy, this.preanim(arguments, 3));
8747             return this;
8748         },
8749
8750         /**
8751          * Anchors an element to another element and realigns it when the window is resized.
8752          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8753          * @param {String} position The position to align to.
8754          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8755          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8756          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8757          * is a number, it is used as the buffer delay (defaults to 50ms).
8758          * @param {Function} callback The function to call after the animation finishes
8759          * @return {Roo.Element} this
8760          */
8761         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8762             var action = function(){
8763                 this.alignTo(el, alignment, offsets, animate);
8764                 Roo.callback(callback, this);
8765             };
8766             Roo.EventManager.onWindowResize(action, this);
8767             var tm = typeof monitorScroll;
8768             if(tm != 'undefined'){
8769                 Roo.EventManager.on(window, 'scroll', action, this,
8770                     {buffer: tm == 'number' ? monitorScroll : 50});
8771             }
8772             action.call(this); // align immediately
8773             return this;
8774         },
8775         /**
8776          * Clears any opacity settings from this element. Required in some cases for IE.
8777          * @return {Roo.Element} this
8778          */
8779         clearOpacity : function(){
8780             if (window.ActiveXObject) {
8781                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8782                     this.dom.style.filter = "";
8783                 }
8784             } else {
8785                 this.dom.style.opacity = "";
8786                 this.dom.style["-moz-opacity"] = "";
8787                 this.dom.style["-khtml-opacity"] = "";
8788             }
8789             return this;
8790         },
8791
8792         /**
8793          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8794          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8795          * @return {Roo.Element} this
8796          */
8797         hide : function(animate){
8798             this.setVisible(false, this.preanim(arguments, 0));
8799             return this;
8800         },
8801
8802         /**
8803         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8804         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8805          * @return {Roo.Element} this
8806          */
8807         show : function(animate){
8808             this.setVisible(true, this.preanim(arguments, 0));
8809             return this;
8810         },
8811
8812         /**
8813          * @private Test if size has a unit, otherwise appends the default
8814          */
8815         addUnits : function(size){
8816             return Roo.Element.addUnits(size, this.defaultUnit);
8817         },
8818
8819         /**
8820          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8821          * @return {Roo.Element} this
8822          */
8823         beginMeasure : function(){
8824             var el = this.dom;
8825             if(el.offsetWidth || el.offsetHeight){
8826                 return this; // offsets work already
8827             }
8828             var changed = [];
8829             var p = this.dom, b = document.body; // start with this element
8830             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8831                 var pe = Roo.get(p);
8832                 if(pe.getStyle('display') == 'none'){
8833                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8834                     p.style.visibility = "hidden";
8835                     p.style.display = "block";
8836                 }
8837                 p = p.parentNode;
8838             }
8839             this._measureChanged = changed;
8840             return this;
8841
8842         },
8843
8844         /**
8845          * Restores displays to before beginMeasure was called
8846          * @return {Roo.Element} this
8847          */
8848         endMeasure : function(){
8849             var changed = this._measureChanged;
8850             if(changed){
8851                 for(var i = 0, len = changed.length; i < len; i++) {
8852                     var r = changed[i];
8853                     r.el.style.visibility = r.visibility;
8854                     r.el.style.display = "none";
8855                 }
8856                 this._measureChanged = null;
8857             }
8858             return this;
8859         },
8860
8861         /**
8862         * Update the innerHTML of this element, optionally searching for and processing scripts
8863         * @param {String} html The new HTML
8864         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8865         * @param {Function} callback For async script loading you can be noticed when the update completes
8866         * @return {Roo.Element} this
8867          */
8868         update : function(html, loadScripts, callback){
8869             if(typeof html == "undefined"){
8870                 html = "";
8871             }
8872             if(loadScripts !== true){
8873                 this.dom.innerHTML = html;
8874                 if(typeof callback == "function"){
8875                     callback();
8876                 }
8877                 return this;
8878             }
8879             var id = Roo.id();
8880             var dom = this.dom;
8881
8882             html += '<span id="' + id + '"></span>';
8883
8884             E.onAvailable(id, function(){
8885                 var hd = document.getElementsByTagName("head")[0];
8886                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8887                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8888                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8889
8890                 var match;
8891                 while(match = re.exec(html)){
8892                     var attrs = match[1];
8893                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8894                     if(srcMatch && srcMatch[2]){
8895                        var s = document.createElement("script");
8896                        s.src = srcMatch[2];
8897                        var typeMatch = attrs.match(typeRe);
8898                        if(typeMatch && typeMatch[2]){
8899                            s.type = typeMatch[2];
8900                        }
8901                        hd.appendChild(s);
8902                     }else if(match[2] && match[2].length > 0){
8903                         if(window.execScript) {
8904                            window.execScript(match[2]);
8905                         } else {
8906                             /**
8907                              * eval:var:id
8908                              * eval:var:dom
8909                              * eval:var:html
8910                              * 
8911                              */
8912                            window.eval(match[2]);
8913                         }
8914                     }
8915                 }
8916                 var el = document.getElementById(id);
8917                 if(el){el.parentNode.removeChild(el);}
8918                 if(typeof callback == "function"){
8919                     callback();
8920                 }
8921             });
8922             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8923             return this;
8924         },
8925
8926         /**
8927          * Direct access to the UpdateManager update() method (takes the same parameters).
8928          * @param {String/Function} url The url for this request or a function to call to get the url
8929          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8930          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8931          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8932          * @return {Roo.Element} this
8933          */
8934         load : function(){
8935             var um = this.getUpdateManager();
8936             um.update.apply(um, arguments);
8937             return this;
8938         },
8939
8940         /**
8941         * Gets this element's UpdateManager
8942         * @return {Roo.UpdateManager} The UpdateManager
8943         */
8944         getUpdateManager : function(){
8945             if(!this.updateManager){
8946                 this.updateManager = new Roo.UpdateManager(this);
8947             }
8948             return this.updateManager;
8949         },
8950
8951         /**
8952          * Disables text selection for this element (normalized across browsers)
8953          * @return {Roo.Element} this
8954          */
8955         unselectable : function(){
8956             this.dom.unselectable = "on";
8957             this.swallowEvent("selectstart", true);
8958             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8959             this.addClass("x-unselectable");
8960             return this;
8961         },
8962
8963         /**
8964         * Calculates the x, y to center this element on the screen
8965         * @return {Array} The x, y values [x, y]
8966         */
8967         getCenterXY : function(){
8968             return this.getAlignToXY(document, 'c-c');
8969         },
8970
8971         /**
8972         * Centers the Element in either the viewport, or another Element.
8973         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8974         */
8975         center : function(centerIn){
8976             this.alignTo(centerIn || document, 'c-c');
8977             return this;
8978         },
8979
8980         /**
8981          * Tests various css rules/browsers to determine if this element uses a border box
8982          * @return {Boolean}
8983          */
8984         isBorderBox : function(){
8985             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8986         },
8987
8988         /**
8989          * Return a box {x, y, width, height} that can be used to set another elements
8990          * size/location to match this element.
8991          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8992          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8993          * @return {Object} box An object in the format {x, y, width, height}
8994          */
8995         getBox : function(contentBox, local){
8996             var xy;
8997             if(!local){
8998                 xy = this.getXY();
8999             }else{
9000                 var left = parseInt(this.getStyle("left"), 10) || 0;
9001                 var top = parseInt(this.getStyle("top"), 10) || 0;
9002                 xy = [left, top];
9003             }
9004             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9005             if(!contentBox){
9006                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9007             }else{
9008                 var l = this.getBorderWidth("l")+this.getPadding("l");
9009                 var r = this.getBorderWidth("r")+this.getPadding("r");
9010                 var t = this.getBorderWidth("t")+this.getPadding("t");
9011                 var b = this.getBorderWidth("b")+this.getPadding("b");
9012                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9013             }
9014             bx.right = bx.x + bx.width;
9015             bx.bottom = bx.y + bx.height;
9016             return bx;
9017         },
9018
9019         /**
9020          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9021          for more information about the sides.
9022          * @param {String} sides
9023          * @return {Number}
9024          */
9025         getFrameWidth : function(sides, onlyContentBox){
9026             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9027         },
9028
9029         /**
9030          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9031          * @param {Object} box The box to fill {x, y, width, height}
9032          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9033          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9034          * @return {Roo.Element} this
9035          */
9036         setBox : function(box, adjust, animate){
9037             var w = box.width, h = box.height;
9038             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9039                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9040                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9041             }
9042             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9043             return this;
9044         },
9045
9046         /**
9047          * Forces the browser to repaint this element
9048          * @return {Roo.Element} this
9049          */
9050          repaint : function(){
9051             var dom = this.dom;
9052             this.addClass("x-repaint");
9053             setTimeout(function(){
9054                 Roo.get(dom).removeClass("x-repaint");
9055             }, 1);
9056             return this;
9057         },
9058
9059         /**
9060          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9061          * then it returns the calculated width of the sides (see getPadding)
9062          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9063          * @return {Object/Number}
9064          */
9065         getMargins : function(side){
9066             if(!side){
9067                 return {
9068                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9069                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9070                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9071                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9072                 };
9073             }else{
9074                 return this.addStyles(side, El.margins);
9075              }
9076         },
9077
9078         // private
9079         addStyles : function(sides, styles){
9080             var val = 0, v, w;
9081             for(var i = 0, len = sides.length; i < len; i++){
9082                 v = this.getStyle(styles[sides.charAt(i)]);
9083                 if(v){
9084                      w = parseInt(v, 10);
9085                      if(w){ val += w; }
9086                 }
9087             }
9088             return val;
9089         },
9090
9091         /**
9092          * Creates a proxy element of this element
9093          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9094          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9095          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9096          * @return {Roo.Element} The new proxy element
9097          */
9098         createProxy : function(config, renderTo, matchBox){
9099             if(renderTo){
9100                 renderTo = Roo.getDom(renderTo);
9101             }else{
9102                 renderTo = document.body;
9103             }
9104             config = typeof config == "object" ?
9105                 config : {tag : "div", cls: config};
9106             var proxy = Roo.DomHelper.append(renderTo, config, true);
9107             if(matchBox){
9108                proxy.setBox(this.getBox());
9109             }
9110             return proxy;
9111         },
9112
9113         /**
9114          * Puts a mask over this element to disable user interaction. Requires core.css.
9115          * This method can only be applied to elements which accept child nodes.
9116          * @param {String} msg (optional) A message to display in the mask
9117          * @param {String} msgCls (optional) A css class to apply to the msg element
9118          * @return {Element} The mask  element
9119          */
9120         mask : function(msg, msgCls)
9121         {
9122             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9123                 this.setStyle("position", "relative");
9124             }
9125             if(!this._mask){
9126                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9127             }
9128             
9129             this.addClass("x-masked");
9130             this._mask.setDisplayed(true);
9131             
9132             // we wander
9133             var z = 0;
9134             var dom = this.dom;
9135             while (dom && dom.style) {
9136                 if (!isNaN(parseInt(dom.style.zIndex))) {
9137                     z = Math.max(z, parseInt(dom.style.zIndex));
9138                 }
9139                 dom = dom.parentNode;
9140             }
9141             // if we are masking the body - then it hides everything..
9142             if (this.dom == document.body) {
9143                 z = 1000000;
9144                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9145                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9146             }
9147            
9148             if(typeof msg == 'string'){
9149                 if(!this._maskMsg){
9150                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9151                         cls: "roo-el-mask-msg", 
9152                         cn: [
9153                             {
9154                                 tag: 'i',
9155                                 cls: 'fa fa-spinner fa-spin'
9156                             },
9157                             {
9158                                 tag: 'div'
9159                             }   
9160                         ]
9161                     }, true);
9162                 }
9163                 var mm = this._maskMsg;
9164                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9165                 if (mm.dom.lastChild) { // weird IE issue?
9166                     mm.dom.lastChild.innerHTML = msg;
9167                 }
9168                 mm.setDisplayed(true);
9169                 mm.center(this);
9170                 mm.setStyle('z-index', z + 102);
9171             }
9172             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9173                 this._mask.setHeight(this.getHeight());
9174             }
9175             this._mask.setStyle('z-index', z + 100);
9176             
9177             return this._mask;
9178         },
9179
9180         /**
9181          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9182          * it is cached for reuse.
9183          */
9184         unmask : function(removeEl){
9185             if(this._mask){
9186                 if(removeEl === true){
9187                     this._mask.remove();
9188                     delete this._mask;
9189                     if(this._maskMsg){
9190                         this._maskMsg.remove();
9191                         delete this._maskMsg;
9192                     }
9193                 }else{
9194                     this._mask.setDisplayed(false);
9195                     if(this._maskMsg){
9196                         this._maskMsg.setDisplayed(false);
9197                     }
9198                 }
9199             }
9200             this.removeClass("x-masked");
9201         },
9202
9203         /**
9204          * Returns true if this element is masked
9205          * @return {Boolean}
9206          */
9207         isMasked : function(){
9208             return this._mask && this._mask.isVisible();
9209         },
9210
9211         /**
9212          * Creates an iframe shim for this element to keep selects and other windowed objects from
9213          * showing through.
9214          * @return {Roo.Element} The new shim element
9215          */
9216         createShim : function(){
9217             var el = document.createElement('iframe');
9218             el.frameBorder = 'no';
9219             el.className = 'roo-shim';
9220             if(Roo.isIE && Roo.isSecure){
9221                 el.src = Roo.SSL_SECURE_URL;
9222             }
9223             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9224             shim.autoBoxAdjust = false;
9225             return shim;
9226         },
9227
9228         /**
9229          * Removes this element from the DOM and deletes it from the cache
9230          */
9231         remove : function(){
9232             if(this.dom.parentNode){
9233                 this.dom.parentNode.removeChild(this.dom);
9234             }
9235             delete El.cache[this.dom.id];
9236         },
9237
9238         /**
9239          * Sets up event handlers to add and remove a css class when the mouse is over this element
9240          * @param {String} className
9241          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9242          * mouseout events for children elements
9243          * @return {Roo.Element} this
9244          */
9245         addClassOnOver : function(className, preventFlicker){
9246             this.on("mouseover", function(){
9247                 Roo.fly(this, '_internal').addClass(className);
9248             }, this.dom);
9249             var removeFn = function(e){
9250                 if(preventFlicker !== true || !e.within(this, true)){
9251                     Roo.fly(this, '_internal').removeClass(className);
9252                 }
9253             };
9254             this.on("mouseout", removeFn, this.dom);
9255             return this;
9256         },
9257
9258         /**
9259          * Sets up event handlers to add and remove a css class when this element has the focus
9260          * @param {String} className
9261          * @return {Roo.Element} this
9262          */
9263         addClassOnFocus : function(className){
9264             this.on("focus", function(){
9265                 Roo.fly(this, '_internal').addClass(className);
9266             }, this.dom);
9267             this.on("blur", function(){
9268                 Roo.fly(this, '_internal').removeClass(className);
9269             }, this.dom);
9270             return this;
9271         },
9272         /**
9273          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9274          * @param {String} className
9275          * @return {Roo.Element} this
9276          */
9277         addClassOnClick : function(className){
9278             var dom = this.dom;
9279             this.on("mousedown", function(){
9280                 Roo.fly(dom, '_internal').addClass(className);
9281                 var d = Roo.get(document);
9282                 var fn = function(){
9283                     Roo.fly(dom, '_internal').removeClass(className);
9284                     d.removeListener("mouseup", fn);
9285                 };
9286                 d.on("mouseup", fn);
9287             });
9288             return this;
9289         },
9290
9291         /**
9292          * Stops the specified event from bubbling and optionally prevents the default action
9293          * @param {String} eventName
9294          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9295          * @return {Roo.Element} this
9296          */
9297         swallowEvent : function(eventName, preventDefault){
9298             var fn = function(e){
9299                 e.stopPropagation();
9300                 if(preventDefault){
9301                     e.preventDefault();
9302                 }
9303             };
9304             if(eventName instanceof Array){
9305                 for(var i = 0, len = eventName.length; i < len; i++){
9306                      this.on(eventName[i], fn);
9307                 }
9308                 return this;
9309             }
9310             this.on(eventName, fn);
9311             return this;
9312         },
9313
9314         /**
9315          * @private
9316          */
9317       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9318
9319         /**
9320          * Sizes this element to its parent element's dimensions performing
9321          * neccessary box adjustments.
9322          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9323          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9324          * @return {Roo.Element} this
9325          */
9326         fitToParent : function(monitorResize, targetParent) {
9327           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9328           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9329           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9330             return;
9331           }
9332           var p = Roo.get(targetParent || this.dom.parentNode);
9333           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9334           if (monitorResize === true) {
9335             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9336             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9337           }
9338           return this;
9339         },
9340
9341         /**
9342          * Gets the next sibling, skipping text nodes
9343          * @return {HTMLElement} The next sibling or null
9344          */
9345         getNextSibling : function(){
9346             var n = this.dom.nextSibling;
9347             while(n && n.nodeType != 1){
9348                 n = n.nextSibling;
9349             }
9350             return n;
9351         },
9352
9353         /**
9354          * Gets the previous sibling, skipping text nodes
9355          * @return {HTMLElement} The previous sibling or null
9356          */
9357         getPrevSibling : function(){
9358             var n = this.dom.previousSibling;
9359             while(n && n.nodeType != 1){
9360                 n = n.previousSibling;
9361             }
9362             return n;
9363         },
9364
9365
9366         /**
9367          * Appends the passed element(s) to this element
9368          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9369          * @return {Roo.Element} this
9370          */
9371         appendChild: function(el){
9372             el = Roo.get(el);
9373             el.appendTo(this);
9374             return this;
9375         },
9376
9377         /**
9378          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9379          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9380          * automatically generated with the specified attributes.
9381          * @param {HTMLElement} insertBefore (optional) a child element of this element
9382          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9383          * @return {Roo.Element} The new child element
9384          */
9385         createChild: function(config, insertBefore, returnDom){
9386             config = config || {tag:'div'};
9387             if(insertBefore){
9388                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9389             }
9390             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9391         },
9392
9393         /**
9394          * Appends this element to the passed element
9395          * @param {String/HTMLElement/Element} el The new parent element
9396          * @return {Roo.Element} this
9397          */
9398         appendTo: function(el){
9399             el = Roo.getDom(el);
9400             el.appendChild(this.dom);
9401             return this;
9402         },
9403
9404         /**
9405          * Inserts this element before the passed element in the DOM
9406          * @param {String/HTMLElement/Element} el The element to insert before
9407          * @return {Roo.Element} this
9408          */
9409         insertBefore: function(el){
9410             el = Roo.getDom(el);
9411             el.parentNode.insertBefore(this.dom, el);
9412             return this;
9413         },
9414
9415         /**
9416          * Inserts this element after the passed element in the DOM
9417          * @param {String/HTMLElement/Element} el The element to insert after
9418          * @return {Roo.Element} this
9419          */
9420         insertAfter: function(el){
9421             el = Roo.getDom(el);
9422             el.parentNode.insertBefore(this.dom, el.nextSibling);
9423             return this;
9424         },
9425
9426         /**
9427          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9428          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9429          * @return {Roo.Element} The new child
9430          */
9431         insertFirst: function(el, returnDom){
9432             el = el || {};
9433             if(typeof el == 'object' && !el.nodeType){ // dh config
9434                 return this.createChild(el, this.dom.firstChild, returnDom);
9435             }else{
9436                 el = Roo.getDom(el);
9437                 this.dom.insertBefore(el, this.dom.firstChild);
9438                 return !returnDom ? Roo.get(el) : el;
9439             }
9440         },
9441
9442         /**
9443          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9444          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9445          * @param {String} where (optional) 'before' or 'after' defaults to before
9446          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9447          * @return {Roo.Element} the inserted Element
9448          */
9449         insertSibling: function(el, where, returnDom){
9450             where = where ? where.toLowerCase() : 'before';
9451             el = el || {};
9452             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9453
9454             if(typeof el == 'object' && !el.nodeType){ // dh config
9455                 if(where == 'after' && !this.dom.nextSibling){
9456                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9457                 }else{
9458                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9459                 }
9460
9461             }else{
9462                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9463                             where == 'before' ? this.dom : this.dom.nextSibling);
9464                 if(!returnDom){
9465                     rt = Roo.get(rt);
9466                 }
9467             }
9468             return rt;
9469         },
9470
9471         /**
9472          * Creates and wraps this element with another element
9473          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9474          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9475          * @return {HTMLElement/Element} The newly created wrapper element
9476          */
9477         wrap: function(config, returnDom){
9478             if(!config){
9479                 config = {tag: "div"};
9480             }
9481             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9482             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9483             return newEl;
9484         },
9485
9486         /**
9487          * Replaces the passed element with this element
9488          * @param {String/HTMLElement/Element} el The element to replace
9489          * @return {Roo.Element} this
9490          */
9491         replace: function(el){
9492             el = Roo.get(el);
9493             this.insertBefore(el);
9494             el.remove();
9495             return this;
9496         },
9497
9498         /**
9499          * Inserts an html fragment into this element
9500          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9501          * @param {String} html The HTML fragment
9502          * @param {Boolean} returnEl True to return an Roo.Element
9503          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9504          */
9505         insertHtml : function(where, html, returnEl){
9506             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9507             return returnEl ? Roo.get(el) : el;
9508         },
9509
9510         /**
9511          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9512          * @param {Object} o The object with the attributes
9513          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9514          * @return {Roo.Element} this
9515          */
9516         set : function(o, useSet){
9517             var el = this.dom;
9518             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9519             for(var attr in o){
9520                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9521                 if(attr=="cls"){
9522                     el.className = o["cls"];
9523                 }else{
9524                     if(useSet) {
9525                         el.setAttribute(attr, o[attr]);
9526                     } else {
9527                         el[attr] = o[attr];
9528                     }
9529                 }
9530             }
9531             if(o.style){
9532                 Roo.DomHelper.applyStyles(el, o.style);
9533             }
9534             return this;
9535         },
9536
9537         /**
9538          * Convenience method for constructing a KeyMap
9539          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9540          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9541          * @param {Function} fn The function to call
9542          * @param {Object} scope (optional) The scope of the function
9543          * @return {Roo.KeyMap} The KeyMap created
9544          */
9545         addKeyListener : function(key, fn, scope){
9546             var config;
9547             if(typeof key != "object" || key instanceof Array){
9548                 config = {
9549                     key: key,
9550                     fn: fn,
9551                     scope: scope
9552                 };
9553             }else{
9554                 config = {
9555                     key : key.key,
9556                     shift : key.shift,
9557                     ctrl : key.ctrl,
9558                     alt : key.alt,
9559                     fn: fn,
9560                     scope: scope
9561                 };
9562             }
9563             return new Roo.KeyMap(this, config);
9564         },
9565
9566         /**
9567          * Creates a KeyMap for this element
9568          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9569          * @return {Roo.KeyMap} The KeyMap created
9570          */
9571         addKeyMap : function(config){
9572             return new Roo.KeyMap(this, config);
9573         },
9574
9575         /**
9576          * Returns true if this element is scrollable.
9577          * @return {Boolean}
9578          */
9579          isScrollable : function(){
9580             var dom = this.dom;
9581             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9582         },
9583
9584         /**
9585          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9586          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9587          * @param {Number} value The new scroll value
9588          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9589          * @return {Element} this
9590          */
9591
9592         scrollTo : function(side, value, animate){
9593             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9594             if(!animate || !A){
9595                 this.dom[prop] = value;
9596             }else{
9597                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9598                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9599             }
9600             return this;
9601         },
9602
9603         /**
9604          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9605          * within this element's scrollable range.
9606          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9607          * @param {Number} distance How far to scroll the element in pixels
9608          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9609          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9610          * was scrolled as far as it could go.
9611          */
9612          scroll : function(direction, distance, animate){
9613              if(!this.isScrollable()){
9614                  return;
9615              }
9616              var el = this.dom;
9617              var l = el.scrollLeft, t = el.scrollTop;
9618              var w = el.scrollWidth, h = el.scrollHeight;
9619              var cw = el.clientWidth, ch = el.clientHeight;
9620              direction = direction.toLowerCase();
9621              var scrolled = false;
9622              var a = this.preanim(arguments, 2);
9623              switch(direction){
9624                  case "l":
9625                  case "left":
9626                      if(w - l > cw){
9627                          var v = Math.min(l + distance, w-cw);
9628                          this.scrollTo("left", v, a);
9629                          scrolled = true;
9630                      }
9631                      break;
9632                 case "r":
9633                 case "right":
9634                      if(l > 0){
9635                          var v = Math.max(l - distance, 0);
9636                          this.scrollTo("left", v, a);
9637                          scrolled = true;
9638                      }
9639                      break;
9640                 case "t":
9641                 case "top":
9642                 case "up":
9643                      if(t > 0){
9644                          var v = Math.max(t - distance, 0);
9645                          this.scrollTo("top", v, a);
9646                          scrolled = true;
9647                      }
9648                      break;
9649                 case "b":
9650                 case "bottom":
9651                 case "down":
9652                      if(h - t > ch){
9653                          var v = Math.min(t + distance, h-ch);
9654                          this.scrollTo("top", v, a);
9655                          scrolled = true;
9656                      }
9657                      break;
9658              }
9659              return scrolled;
9660         },
9661
9662         /**
9663          * Translates the passed page coordinates into left/top css values for this element
9664          * @param {Number/Array} x The page x or an array containing [x, y]
9665          * @param {Number} y The page y
9666          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9667          */
9668         translatePoints : function(x, y){
9669             if(typeof x == 'object' || x instanceof Array){
9670                 y = x[1]; x = x[0];
9671             }
9672             var p = this.getStyle('position');
9673             var o = this.getXY();
9674
9675             var l = parseInt(this.getStyle('left'), 10);
9676             var t = parseInt(this.getStyle('top'), 10);
9677
9678             if(isNaN(l)){
9679                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9680             }
9681             if(isNaN(t)){
9682                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9683             }
9684
9685             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9686         },
9687
9688         /**
9689          * Returns the current scroll position of the element.
9690          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9691          */
9692         getScroll : function(){
9693             var d = this.dom, doc = document;
9694             if(d == doc || d == doc.body){
9695                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9696                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9697                 return {left: l, top: t};
9698             }else{
9699                 return {left: d.scrollLeft, top: d.scrollTop};
9700             }
9701         },
9702
9703         /**
9704          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9705          * are convert to standard 6 digit hex color.
9706          * @param {String} attr The css attribute
9707          * @param {String} defaultValue The default value to use when a valid color isn't found
9708          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9709          * YUI color anims.
9710          */
9711         getColor : function(attr, defaultValue, prefix){
9712             var v = this.getStyle(attr);
9713             if(!v || v == "transparent" || v == "inherit") {
9714                 return defaultValue;
9715             }
9716             var color = typeof prefix == "undefined" ? "#" : prefix;
9717             if(v.substr(0, 4) == "rgb("){
9718                 var rvs = v.slice(4, v.length -1).split(",");
9719                 for(var i = 0; i < 3; i++){
9720                     var h = parseInt(rvs[i]).toString(16);
9721                     if(h < 16){
9722                         h = "0" + h;
9723                     }
9724                     color += h;
9725                 }
9726             } else {
9727                 if(v.substr(0, 1) == "#"){
9728                     if(v.length == 4) {
9729                         for(var i = 1; i < 4; i++){
9730                             var c = v.charAt(i);
9731                             color +=  c + c;
9732                         }
9733                     }else if(v.length == 7){
9734                         color += v.substr(1);
9735                     }
9736                 }
9737             }
9738             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9739         },
9740
9741         /**
9742          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9743          * gradient background, rounded corners and a 4-way shadow.
9744          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9745          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9746          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9747          * @return {Roo.Element} this
9748          */
9749         boxWrap : function(cls){
9750             cls = cls || 'x-box';
9751             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9752             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9753             return el;
9754         },
9755
9756         /**
9757          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9758          * @param {String} namespace The namespace in which to look for the attribute
9759          * @param {String} name The attribute name
9760          * @return {String} The attribute value
9761          */
9762         getAttributeNS : Roo.isIE ? function(ns, name){
9763             var d = this.dom;
9764             var type = typeof d[ns+":"+name];
9765             if(type != 'undefined' && type != 'unknown'){
9766                 return d[ns+":"+name];
9767             }
9768             return d[name];
9769         } : function(ns, name){
9770             var d = this.dom;
9771             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9772         },
9773         
9774         
9775         /**
9776          * Sets or Returns the value the dom attribute value
9777          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9778          * @param {String} value (optional) The value to set the attribute to
9779          * @return {String} The attribute value
9780          */
9781         attr : function(name){
9782             if (arguments.length > 1) {
9783                 this.dom.setAttribute(name, arguments[1]);
9784                 return arguments[1];
9785             }
9786             if (typeof(name) == 'object') {
9787                 for(var i in name) {
9788                     this.attr(i, name[i]);
9789                 }
9790                 return name;
9791             }
9792             
9793             
9794             if (!this.dom.hasAttribute(name)) {
9795                 return undefined;
9796             }
9797             return this.dom.getAttribute(name);
9798         }
9799         
9800         
9801         
9802     };
9803
9804     var ep = El.prototype;
9805
9806     /**
9807      * Appends an event handler (Shorthand for addListener)
9808      * @param {String}   eventName     The type of event to append
9809      * @param {Function} fn        The method the event invokes
9810      * @param {Object} scope       (optional) The scope (this object) of the fn
9811      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9812      * @method
9813      */
9814     ep.on = ep.addListener;
9815         // backwards compat
9816     ep.mon = ep.addListener;
9817
9818     /**
9819      * Removes an event handler from this element (shorthand for removeListener)
9820      * @param {String} eventName the type of event to remove
9821      * @param {Function} fn the method the event invokes
9822      * @return {Roo.Element} this
9823      * @method
9824      */
9825     ep.un = ep.removeListener;
9826
9827     /**
9828      * true to automatically adjust width and height settings for box-model issues (default to true)
9829      */
9830     ep.autoBoxAdjust = true;
9831
9832     // private
9833     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9834
9835     // private
9836     El.addUnits = function(v, defaultUnit){
9837         if(v === "" || v == "auto"){
9838             return v;
9839         }
9840         if(v === undefined){
9841             return '';
9842         }
9843         if(typeof v == "number" || !El.unitPattern.test(v)){
9844             return v + (defaultUnit || 'px');
9845         }
9846         return v;
9847     };
9848
9849     // special markup used throughout Roo when box wrapping elements
9850     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9851     /**
9852      * Visibility mode constant - Use visibility to hide element
9853      * @static
9854      * @type Number
9855      */
9856     El.VISIBILITY = 1;
9857     /**
9858      * Visibility mode constant - Use display to hide element
9859      * @static
9860      * @type Number
9861      */
9862     El.DISPLAY = 2;
9863
9864     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9865     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9866     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9867
9868
9869
9870     /**
9871      * @private
9872      */
9873     El.cache = {};
9874
9875     var docEl;
9876
9877     /**
9878      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9879      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9880      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9881      * @return {Element} The Element object
9882      * @static
9883      */
9884     El.get = function(el){
9885         var ex, elm, id;
9886         if(!el){ return null; }
9887         if(typeof el == "string"){ // element id
9888             if(!(elm = document.getElementById(el))){
9889                 return null;
9890             }
9891             if(ex = El.cache[el]){
9892                 ex.dom = elm;
9893             }else{
9894                 ex = El.cache[el] = new El(elm);
9895             }
9896             return ex;
9897         }else if(el.tagName){ // dom element
9898             if(!(id = el.id)){
9899                 id = Roo.id(el);
9900             }
9901             if(ex = El.cache[id]){
9902                 ex.dom = el;
9903             }else{
9904                 ex = El.cache[id] = new El(el);
9905             }
9906             return ex;
9907         }else if(el instanceof El){
9908             if(el != docEl){
9909                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9910                                                               // catch case where it hasn't been appended
9911                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9912             }
9913             return el;
9914         }else if(el.isComposite){
9915             return el;
9916         }else if(el instanceof Array){
9917             return El.select(el);
9918         }else if(el == document){
9919             // create a bogus element object representing the document object
9920             if(!docEl){
9921                 var f = function(){};
9922                 f.prototype = El.prototype;
9923                 docEl = new f();
9924                 docEl.dom = document;
9925             }
9926             return docEl;
9927         }
9928         return null;
9929     };
9930
9931     // private
9932     El.uncache = function(el){
9933         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9934             if(a[i]){
9935                 delete El.cache[a[i].id || a[i]];
9936             }
9937         }
9938     };
9939
9940     // private
9941     // Garbage collection - uncache elements/purge listeners on orphaned elements
9942     // so we don't hold a reference and cause the browser to retain them
9943     El.garbageCollect = function(){
9944         if(!Roo.enableGarbageCollector){
9945             clearInterval(El.collectorThread);
9946             return;
9947         }
9948         for(var eid in El.cache){
9949             var el = El.cache[eid], d = el.dom;
9950             // -------------------------------------------------------
9951             // Determining what is garbage:
9952             // -------------------------------------------------------
9953             // !d
9954             // dom node is null, definitely garbage
9955             // -------------------------------------------------------
9956             // !d.parentNode
9957             // no parentNode == direct orphan, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.offsetParent && !document.getElementById(eid)
9960             // display none elements have no offsetParent so we will
9961             // also try to look it up by it's id. However, check
9962             // offsetParent first so we don't do unneeded lookups.
9963             // This enables collection of elements that are not orphans
9964             // directly, but somewhere up the line they have an orphan
9965             // parent.
9966             // -------------------------------------------------------
9967             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9968                 delete El.cache[eid];
9969                 if(d && Roo.enableListenerCollection){
9970                     E.purgeElement(d);
9971                 }
9972             }
9973         }
9974     }
9975     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9976
9977
9978     // dom is optional
9979     El.Flyweight = function(dom){
9980         this.dom = dom;
9981     };
9982     El.Flyweight.prototype = El.prototype;
9983
9984     El._flyweights = {};
9985     /**
9986      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9987      * the dom node can be overwritten by other code.
9988      * @param {String/HTMLElement} el The dom node or id
9989      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9990      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9991      * @static
9992      * @return {Element} The shared Element object
9993      */
9994     El.fly = function(el, named){
9995         named = named || '_global';
9996         el = Roo.getDom(el);
9997         if(!el){
9998             return null;
9999         }
10000         if(!El._flyweights[named]){
10001             El._flyweights[named] = new El.Flyweight();
10002         }
10003         El._flyweights[named].dom = el;
10004         return El._flyweights[named];
10005     };
10006
10007     /**
10008      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10009      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10010      * Shorthand of {@link Roo.Element#get}
10011      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10012      * @return {Element} The Element object
10013      * @member Roo
10014      * @method get
10015      */
10016     Roo.get = El.get;
10017     /**
10018      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10019      * the dom node can be overwritten by other code.
10020      * Shorthand of {@link Roo.Element#fly}
10021      * @param {String/HTMLElement} el The dom node or id
10022      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10023      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10024      * @static
10025      * @return {Element} The shared Element object
10026      * @member Roo
10027      * @method fly
10028      */
10029     Roo.fly = El.fly;
10030
10031     // speedy lookup for elements never to box adjust
10032     var noBoxAdjust = Roo.isStrict ? {
10033         select:1
10034     } : {
10035         input:1, select:1, textarea:1
10036     };
10037     if(Roo.isIE || Roo.isGecko){
10038         noBoxAdjust['button'] = 1;
10039     }
10040
10041
10042     Roo.EventManager.on(window, 'unload', function(){
10043         delete El.cache;
10044         delete El._flyweights;
10045     });
10046 })();
10047
10048
10049
10050
10051 if(Roo.DomQuery){
10052     Roo.Element.selectorFunction = Roo.DomQuery.select;
10053 }
10054
10055 Roo.Element.select = function(selector, unique, root){
10056     var els;
10057     if(typeof selector == "string"){
10058         els = Roo.Element.selectorFunction(selector, root);
10059     }else if(selector.length !== undefined){
10060         els = selector;
10061     }else{
10062         throw "Invalid selector";
10063     }
10064     if(unique === true){
10065         return new Roo.CompositeElement(els);
10066     }else{
10067         return new Roo.CompositeElementLite(els);
10068     }
10069 };
10070 /**
10071  * Selects elements based on the passed CSS selector to enable working on them as 1.
10072  * @param {String/Array} selector The CSS selector or an array of elements
10073  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10074  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10075  * @return {CompositeElementLite/CompositeElement}
10076  * @member Roo
10077  * @method select
10078  */
10079 Roo.select = Roo.Element.select;
10080
10081
10082
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094 /*
10095  * Based on:
10096  * Ext JS Library 1.1.1
10097  * Copyright(c) 2006-2007, Ext JS, LLC.
10098  *
10099  * Originally Released Under LGPL - original licence link has changed is not relivant.
10100  *
10101  * Fork - LGPL
10102  * <script type="text/javascript">
10103  */
10104
10105
10106
10107 //Notifies Element that fx methods are available
10108 Roo.enableFx = true;
10109
10110 /**
10111  * @class Roo.Fx
10112  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10113  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10114  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10115  * Element effects to work.</p><br/>
10116  *
10117  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10118  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10119  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10120  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10121  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10122  * expected results and should be done with care.</p><br/>
10123  *
10124  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10125  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10126 <pre>
10127 Value  Description
10128 -----  -----------------------------
10129 tl     The top left corner
10130 t      The center of the top edge
10131 tr     The top right corner
10132 l      The center of the left edge
10133 r      The center of the right edge
10134 bl     The bottom left corner
10135 b      The center of the bottom edge
10136 br     The bottom right corner
10137 </pre>
10138  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10139  * below are common options that can be passed to any Fx method.</b>
10140  * @cfg {Function} callback A function called when the effect is finished
10141  * @cfg {Object} scope The scope of the effect function
10142  * @cfg {String} easing A valid Easing value for the effect
10143  * @cfg {String} afterCls A css class to apply after the effect
10144  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10145  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10146  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10147  * effects that end with the element being visually hidden, ignored otherwise)
10148  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10149  * a function which returns such a specification that will be applied to the Element after the effect finishes
10150  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10151  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10152  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10153  */
10154 Roo.Fx = {
10155         /**
10156          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10157          * origin for the slide effect.  This function automatically handles wrapping the element with
10158          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10159          * Usage:
10160          *<pre><code>
10161 // default: slide the element in from the top
10162 el.slideIn();
10163
10164 // custom: slide the element in from the right with a 2-second duration
10165 el.slideIn('r', { duration: 2 });
10166
10167 // common config options shown with default values
10168 el.slideIn('t', {
10169     easing: 'easeOut',
10170     duration: .5
10171 });
10172 </code></pre>
10173          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10174          * @param {Object} options (optional) Object literal with any of the Fx config options
10175          * @return {Roo.Element} The Element
10176          */
10177     slideIn : function(anchor, o){
10178         var el = this.getFxEl();
10179         o = o || {};
10180
10181         el.queueFx(o, function(){
10182
10183             anchor = anchor || "t";
10184
10185             // fix display to visibility
10186             this.fixDisplay();
10187
10188             // restore values after effect
10189             var r = this.getFxRestore();
10190             var b = this.getBox();
10191             // fixed size for slide
10192             this.setSize(b);
10193
10194             // wrap if needed
10195             var wrap = this.fxWrap(r.pos, o, "hidden");
10196
10197             var st = this.dom.style;
10198             st.visibility = "visible";
10199             st.position = "absolute";
10200
10201             // clear out temp styles after slide and unwrap
10202             var after = function(){
10203                 el.fxUnwrap(wrap, r.pos, o);
10204                 st.width = r.width;
10205                 st.height = r.height;
10206                 el.afterFx(o);
10207             };
10208             // time to calc the positions
10209             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10210
10211             switch(anchor.toLowerCase()){
10212                 case "t":
10213                     wrap.setSize(b.width, 0);
10214                     st.left = st.bottom = "0";
10215                     a = {height: bh};
10216                 break;
10217                 case "l":
10218                     wrap.setSize(0, b.height);
10219                     st.right = st.top = "0";
10220                     a = {width: bw};
10221                 break;
10222                 case "r":
10223                     wrap.setSize(0, b.height);
10224                     wrap.setX(b.right);
10225                     st.left = st.top = "0";
10226                     a = {width: bw, points: pt};
10227                 break;
10228                 case "b":
10229                     wrap.setSize(b.width, 0);
10230                     wrap.setY(b.bottom);
10231                     st.left = st.top = "0";
10232                     a = {height: bh, points: pt};
10233                 break;
10234                 case "tl":
10235                     wrap.setSize(0, 0);
10236                     st.right = st.bottom = "0";
10237                     a = {width: bw, height: bh};
10238                 break;
10239                 case "bl":
10240                     wrap.setSize(0, 0);
10241                     wrap.setY(b.y+b.height);
10242                     st.right = st.top = "0";
10243                     a = {width: bw, height: bh, points: pt};
10244                 break;
10245                 case "br":
10246                     wrap.setSize(0, 0);
10247                     wrap.setXY([b.right, b.bottom]);
10248                     st.left = st.top = "0";
10249                     a = {width: bw, height: bh, points: pt};
10250                 break;
10251                 case "tr":
10252                     wrap.setSize(0, 0);
10253                     wrap.setX(b.x+b.width);
10254                     st.left = st.bottom = "0";
10255                     a = {width: bw, height: bh, points: pt};
10256                 break;
10257             }
10258             this.dom.style.visibility = "visible";
10259             wrap.show();
10260
10261             arguments.callee.anim = wrap.fxanim(a,
10262                 o,
10263                 'motion',
10264                 .5,
10265                 'easeOut', after);
10266         });
10267         return this;
10268     },
10269     
10270         /**
10271          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10272          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10273          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10274          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10275          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10276          * Usage:
10277          *<pre><code>
10278 // default: slide the element out to the top
10279 el.slideOut();
10280
10281 // custom: slide the element out to the right with a 2-second duration
10282 el.slideOut('r', { duration: 2 });
10283
10284 // common config options shown with default values
10285 el.slideOut('t', {
10286     easing: 'easeOut',
10287     duration: .5,
10288     remove: false,
10289     useDisplay: false
10290 });
10291 </code></pre>
10292          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10293          * @param {Object} options (optional) Object literal with any of the Fx config options
10294          * @return {Roo.Element} The Element
10295          */
10296     slideOut : function(anchor, o){
10297         var el = this.getFxEl();
10298         o = o || {};
10299
10300         el.queueFx(o, function(){
10301
10302             anchor = anchor || "t";
10303
10304             // restore values after effect
10305             var r = this.getFxRestore();
10306             
10307             var b = this.getBox();
10308             // fixed size for slide
10309             this.setSize(b);
10310
10311             // wrap if needed
10312             var wrap = this.fxWrap(r.pos, o, "visible");
10313
10314             var st = this.dom.style;
10315             st.visibility = "visible";
10316             st.position = "absolute";
10317
10318             wrap.setSize(b);
10319
10320             var after = function(){
10321                 if(o.useDisplay){
10322                     el.setDisplayed(false);
10323                 }else{
10324                     el.hide();
10325                 }
10326
10327                 el.fxUnwrap(wrap, r.pos, o);
10328
10329                 st.width = r.width;
10330                 st.height = r.height;
10331
10332                 el.afterFx(o);
10333             };
10334
10335             var a, zero = {to: 0};
10336             switch(anchor.toLowerCase()){
10337                 case "t":
10338                     st.left = st.bottom = "0";
10339                     a = {height: zero};
10340                 break;
10341                 case "l":
10342                     st.right = st.top = "0";
10343                     a = {width: zero};
10344                 break;
10345                 case "r":
10346                     st.left = st.top = "0";
10347                     a = {width: zero, points: {to:[b.right, b.y]}};
10348                 break;
10349                 case "b":
10350                     st.left = st.top = "0";
10351                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10352                 break;
10353                 case "tl":
10354                     st.right = st.bottom = "0";
10355                     a = {width: zero, height: zero};
10356                 break;
10357                 case "bl":
10358                     st.right = st.top = "0";
10359                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10360                 break;
10361                 case "br":
10362                     st.left = st.top = "0";
10363                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10364                 break;
10365                 case "tr":
10366                     st.left = st.bottom = "0";
10367                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10368                 break;
10369             }
10370
10371             arguments.callee.anim = wrap.fxanim(a,
10372                 o,
10373                 'motion',
10374                 .5,
10375                 "easeOut", after);
10376         });
10377         return this;
10378     },
10379
10380         /**
10381          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10382          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10383          * The element must be removed from the DOM using the 'remove' config option if desired.
10384          * Usage:
10385          *<pre><code>
10386 // default
10387 el.puff();
10388
10389 // common config options shown with default values
10390 el.puff({
10391     easing: 'easeOut',
10392     duration: .5,
10393     remove: false,
10394     useDisplay: false
10395 });
10396 </code></pre>
10397          * @param {Object} options (optional) Object literal with any of the Fx config options
10398          * @return {Roo.Element} The Element
10399          */
10400     puff : function(o){
10401         var el = this.getFxEl();
10402         o = o || {};
10403
10404         el.queueFx(o, function(){
10405             this.clearOpacity();
10406             this.show();
10407
10408             // restore values after effect
10409             var r = this.getFxRestore();
10410             var st = this.dom.style;
10411
10412             var after = function(){
10413                 if(o.useDisplay){
10414                     el.setDisplayed(false);
10415                 }else{
10416                     el.hide();
10417                 }
10418
10419                 el.clearOpacity();
10420
10421                 el.setPositioning(r.pos);
10422                 st.width = r.width;
10423                 st.height = r.height;
10424                 st.fontSize = '';
10425                 el.afterFx(o);
10426             };
10427
10428             var width = this.getWidth();
10429             var height = this.getHeight();
10430
10431             arguments.callee.anim = this.fxanim({
10432                     width : {to: this.adjustWidth(width * 2)},
10433                     height : {to: this.adjustHeight(height * 2)},
10434                     points : {by: [-(width * .5), -(height * .5)]},
10435                     opacity : {to: 0},
10436                     fontSize: {to:200, unit: "%"}
10437                 },
10438                 o,
10439                 'motion',
10440                 .5,
10441                 "easeOut", after);
10442         });
10443         return this;
10444     },
10445
10446         /**
10447          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10448          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10449          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10450          * Usage:
10451          *<pre><code>
10452 // default
10453 el.switchOff();
10454
10455 // all config options shown with default values
10456 el.switchOff({
10457     easing: 'easeIn',
10458     duration: .3,
10459     remove: false,
10460     useDisplay: false
10461 });
10462 </code></pre>
10463          * @param {Object} options (optional) Object literal with any of the Fx config options
10464          * @return {Roo.Element} The Element
10465          */
10466     switchOff : function(o){
10467         var el = this.getFxEl();
10468         o = o || {};
10469
10470         el.queueFx(o, function(){
10471             this.clearOpacity();
10472             this.clip();
10473
10474             // restore values after effect
10475             var r = this.getFxRestore();
10476             var st = this.dom.style;
10477
10478             var after = function(){
10479                 if(o.useDisplay){
10480                     el.setDisplayed(false);
10481                 }else{
10482                     el.hide();
10483                 }
10484
10485                 el.clearOpacity();
10486                 el.setPositioning(r.pos);
10487                 st.width = r.width;
10488                 st.height = r.height;
10489
10490                 el.afterFx(o);
10491             };
10492
10493             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10494                 this.clearOpacity();
10495                 (function(){
10496                     this.fxanim({
10497                         height:{to:1},
10498                         points:{by:[0, this.getHeight() * .5]}
10499                     }, o, 'motion', 0.3, 'easeIn', after);
10500                 }).defer(100, this);
10501             });
10502         });
10503         return this;
10504     },
10505
10506     /**
10507      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10508      * changed using the "attr" config option) and then fading back to the original color. If no original
10509      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10510      * Usage:
10511 <pre><code>
10512 // default: highlight background to yellow
10513 el.highlight();
10514
10515 // custom: highlight foreground text to blue for 2 seconds
10516 el.highlight("0000ff", { attr: 'color', duration: 2 });
10517
10518 // common config options shown with default values
10519 el.highlight("ffff9c", {
10520     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10521     endColor: (current color) or "ffffff",
10522     easing: 'easeIn',
10523     duration: 1
10524 });
10525 </code></pre>
10526      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10527      * @param {Object} options (optional) Object literal with any of the Fx config options
10528      * @return {Roo.Element} The Element
10529      */ 
10530     highlight : function(color, o){
10531         var el = this.getFxEl();
10532         o = o || {};
10533
10534         el.queueFx(o, function(){
10535             color = color || "ffff9c";
10536             attr = o.attr || "backgroundColor";
10537
10538             this.clearOpacity();
10539             this.show();
10540
10541             var origColor = this.getColor(attr);
10542             var restoreColor = this.dom.style[attr];
10543             endColor = (o.endColor || origColor) || "ffffff";
10544
10545             var after = function(){
10546                 el.dom.style[attr] = restoreColor;
10547                 el.afterFx(o);
10548             };
10549
10550             var a = {};
10551             a[attr] = {from: color, to: endColor};
10552             arguments.callee.anim = this.fxanim(a,
10553                 o,
10554                 'color',
10555                 1,
10556                 'easeIn', after);
10557         });
10558         return this;
10559     },
10560
10561    /**
10562     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10563     * Usage:
10564 <pre><code>
10565 // default: a single light blue ripple
10566 el.frame();
10567
10568 // custom: 3 red ripples lasting 3 seconds total
10569 el.frame("ff0000", 3, { duration: 3 });
10570
10571 // common config options shown with default values
10572 el.frame("C3DAF9", 1, {
10573     duration: 1 //duration of entire animation (not each individual ripple)
10574     // Note: Easing is not configurable and will be ignored if included
10575 });
10576 </code></pre>
10577     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10578     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10579     * @param {Object} options (optional) Object literal with any of the Fx config options
10580     * @return {Roo.Element} The Element
10581     */
10582     frame : function(color, count, o){
10583         var el = this.getFxEl();
10584         o = o || {};
10585
10586         el.queueFx(o, function(){
10587             color = color || "#C3DAF9";
10588             if(color.length == 6){
10589                 color = "#" + color;
10590             }
10591             count = count || 1;
10592             duration = o.duration || 1;
10593             this.show();
10594
10595             var b = this.getBox();
10596             var animFn = function(){
10597                 var proxy = this.createProxy({
10598
10599                      style:{
10600                         visbility:"hidden",
10601                         position:"absolute",
10602                         "z-index":"35000", // yee haw
10603                         border:"0px solid " + color
10604                      }
10605                   });
10606                 var scale = Roo.isBorderBox ? 2 : 1;
10607                 proxy.animate({
10608                     top:{from:b.y, to:b.y - 20},
10609                     left:{from:b.x, to:b.x - 20},
10610                     borderWidth:{from:0, to:10},
10611                     opacity:{from:1, to:0},
10612                     height:{from:b.height, to:(b.height + (20*scale))},
10613                     width:{from:b.width, to:(b.width + (20*scale))}
10614                 }, duration, function(){
10615                     proxy.remove();
10616                 });
10617                 if(--count > 0){
10618                      animFn.defer((duration/2)*1000, this);
10619                 }else{
10620                     el.afterFx(o);
10621                 }
10622             };
10623             animFn.call(this);
10624         });
10625         return this;
10626     },
10627
10628    /**
10629     * Creates a pause before any subsequent queued effects begin.  If there are
10630     * no effects queued after the pause it will have no effect.
10631     * Usage:
10632 <pre><code>
10633 el.pause(1);
10634 </code></pre>
10635     * @param {Number} seconds The length of time to pause (in seconds)
10636     * @return {Roo.Element} The Element
10637     */
10638     pause : function(seconds){
10639         var el = this.getFxEl();
10640         var o = {};
10641
10642         el.queueFx(o, function(){
10643             setTimeout(function(){
10644                 el.afterFx(o);
10645             }, seconds * 1000);
10646         });
10647         return this;
10648     },
10649
10650    /**
10651     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10652     * using the "endOpacity" config option.
10653     * Usage:
10654 <pre><code>
10655 // default: fade in from opacity 0 to 100%
10656 el.fadeIn();
10657
10658 // custom: fade in from opacity 0 to 75% over 2 seconds
10659 el.fadeIn({ endOpacity: .75, duration: 2});
10660
10661 // common config options shown with default values
10662 el.fadeIn({
10663     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10664     easing: 'easeOut',
10665     duration: .5
10666 });
10667 </code></pre>
10668     * @param {Object} options (optional) Object literal with any of the Fx config options
10669     * @return {Roo.Element} The Element
10670     */
10671     fadeIn : function(o){
10672         var el = this.getFxEl();
10673         o = o || {};
10674         el.queueFx(o, function(){
10675             this.setOpacity(0);
10676             this.fixDisplay();
10677             this.dom.style.visibility = 'visible';
10678             var to = o.endOpacity || 1;
10679             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10680                 o, null, .5, "easeOut", function(){
10681                 if(to == 1){
10682                     this.clearOpacity();
10683                 }
10684                 el.afterFx(o);
10685             });
10686         });
10687         return this;
10688     },
10689
10690    /**
10691     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10692     * using the "endOpacity" config option.
10693     * Usage:
10694 <pre><code>
10695 // default: fade out from the element's current opacity to 0
10696 el.fadeOut();
10697
10698 // custom: fade out from the element's current opacity to 25% over 2 seconds
10699 el.fadeOut({ endOpacity: .25, duration: 2});
10700
10701 // common config options shown with default values
10702 el.fadeOut({
10703     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10704     easing: 'easeOut',
10705     duration: .5
10706     remove: false,
10707     useDisplay: false
10708 });
10709 </code></pre>
10710     * @param {Object} options (optional) Object literal with any of the Fx config options
10711     * @return {Roo.Element} The Element
10712     */
10713     fadeOut : function(o){
10714         var el = this.getFxEl();
10715         o = o || {};
10716         el.queueFx(o, function(){
10717             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10718                 o, null, .5, "easeOut", function(){
10719                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10720                      this.dom.style.display = "none";
10721                 }else{
10722                      this.dom.style.visibility = "hidden";
10723                 }
10724                 this.clearOpacity();
10725                 el.afterFx(o);
10726             });
10727         });
10728         return this;
10729     },
10730
10731    /**
10732     * Animates the transition of an element's dimensions from a starting height/width
10733     * to an ending height/width.
10734     * Usage:
10735 <pre><code>
10736 // change height and width to 100x100 pixels
10737 el.scale(100, 100);
10738
10739 // common config options shown with default values.  The height and width will default to
10740 // the element's existing values if passed as null.
10741 el.scale(
10742     [element's width],
10743     [element's height], {
10744     easing: 'easeOut',
10745     duration: .35
10746 });
10747 </code></pre>
10748     * @param {Number} width  The new width (pass undefined to keep the original width)
10749     * @param {Number} height  The new height (pass undefined to keep the original height)
10750     * @param {Object} options (optional) Object literal with any of the Fx config options
10751     * @return {Roo.Element} The Element
10752     */
10753     scale : function(w, h, o){
10754         this.shift(Roo.apply({}, o, {
10755             width: w,
10756             height: h
10757         }));
10758         return this;
10759     },
10760
10761    /**
10762     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10763     * Any of these properties not specified in the config object will not be changed.  This effect 
10764     * requires that at least one new dimension, position or opacity setting must be passed in on
10765     * the config object in order for the function to have any effect.
10766     * Usage:
10767 <pre><code>
10768 // slide the element horizontally to x position 200 while changing the height and opacity
10769 el.shift({ x: 200, height: 50, opacity: .8 });
10770
10771 // common config options shown with default values.
10772 el.shift({
10773     width: [element's width],
10774     height: [element's height],
10775     x: [element's x position],
10776     y: [element's y position],
10777     opacity: [element's opacity],
10778     easing: 'easeOut',
10779     duration: .35
10780 });
10781 </code></pre>
10782     * @param {Object} options  Object literal with any of the Fx config options
10783     * @return {Roo.Element} The Element
10784     */
10785     shift : function(o){
10786         var el = this.getFxEl();
10787         o = o || {};
10788         el.queueFx(o, function(){
10789             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10790             if(w !== undefined){
10791                 a.width = {to: this.adjustWidth(w)};
10792             }
10793             if(h !== undefined){
10794                 a.height = {to: this.adjustHeight(h)};
10795             }
10796             if(x !== undefined || y !== undefined){
10797                 a.points = {to: [
10798                     x !== undefined ? x : this.getX(),
10799                     y !== undefined ? y : this.getY()
10800                 ]};
10801             }
10802             if(op !== undefined){
10803                 a.opacity = {to: op};
10804             }
10805             if(o.xy !== undefined){
10806                 a.points = {to: o.xy};
10807             }
10808             arguments.callee.anim = this.fxanim(a,
10809                 o, 'motion', .35, "easeOut", function(){
10810                 el.afterFx(o);
10811             });
10812         });
10813         return this;
10814     },
10815
10816         /**
10817          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10818          * ending point of the effect.
10819          * Usage:
10820          *<pre><code>
10821 // default: slide the element downward while fading out
10822 el.ghost();
10823
10824 // custom: slide the element out to the right with a 2-second duration
10825 el.ghost('r', { duration: 2 });
10826
10827 // common config options shown with default values
10828 el.ghost('b', {
10829     easing: 'easeOut',
10830     duration: .5
10831     remove: false,
10832     useDisplay: false
10833 });
10834 </code></pre>
10835          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10836          * @param {Object} options (optional) Object literal with any of the Fx config options
10837          * @return {Roo.Element} The Element
10838          */
10839     ghost : function(anchor, o){
10840         var el = this.getFxEl();
10841         o = o || {};
10842
10843         el.queueFx(o, function(){
10844             anchor = anchor || "b";
10845
10846             // restore values after effect
10847             var r = this.getFxRestore();
10848             var w = this.getWidth(),
10849                 h = this.getHeight();
10850
10851             var st = this.dom.style;
10852
10853             var after = function(){
10854                 if(o.useDisplay){
10855                     el.setDisplayed(false);
10856                 }else{
10857                     el.hide();
10858                 }
10859
10860                 el.clearOpacity();
10861                 el.setPositioning(r.pos);
10862                 st.width = r.width;
10863                 st.height = r.height;
10864
10865                 el.afterFx(o);
10866             };
10867
10868             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10869             switch(anchor.toLowerCase()){
10870                 case "t":
10871                     pt.by = [0, -h];
10872                 break;
10873                 case "l":
10874                     pt.by = [-w, 0];
10875                 break;
10876                 case "r":
10877                     pt.by = [w, 0];
10878                 break;
10879                 case "b":
10880                     pt.by = [0, h];
10881                 break;
10882                 case "tl":
10883                     pt.by = [-w, -h];
10884                 break;
10885                 case "bl":
10886                     pt.by = [-w, h];
10887                 break;
10888                 case "br":
10889                     pt.by = [w, h];
10890                 break;
10891                 case "tr":
10892                     pt.by = [w, -h];
10893                 break;
10894             }
10895
10896             arguments.callee.anim = this.fxanim(a,
10897                 o,
10898                 'motion',
10899                 .5,
10900                 "easeOut", after);
10901         });
10902         return this;
10903     },
10904
10905         /**
10906          * Ensures that all effects queued after syncFx is called on the element are
10907          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10908          * @return {Roo.Element} The Element
10909          */
10910     syncFx : function(){
10911         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10912             block : false,
10913             concurrent : true,
10914             stopFx : false
10915         });
10916         return this;
10917     },
10918
10919         /**
10920          * Ensures that all effects queued after sequenceFx is called on the element are
10921          * run in sequence.  This is the opposite of {@link #syncFx}.
10922          * @return {Roo.Element} The Element
10923          */
10924     sequenceFx : function(){
10925         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10926             block : false,
10927             concurrent : false,
10928             stopFx : false
10929         });
10930         return this;
10931     },
10932
10933         /* @private */
10934     nextFx : function(){
10935         var ef = this.fxQueue[0];
10936         if(ef){
10937             ef.call(this);
10938         }
10939     },
10940
10941         /**
10942          * Returns true if the element has any effects actively running or queued, else returns false.
10943          * @return {Boolean} True if element has active effects, else false
10944          */
10945     hasActiveFx : function(){
10946         return this.fxQueue && this.fxQueue[0];
10947     },
10948
10949         /**
10950          * Stops any running effects and clears the element's internal effects queue if it contains
10951          * any additional effects that haven't started yet.
10952          * @return {Roo.Element} The Element
10953          */
10954     stopFx : function(){
10955         if(this.hasActiveFx()){
10956             var cur = this.fxQueue[0];
10957             if(cur && cur.anim && cur.anim.isAnimated()){
10958                 this.fxQueue = [cur]; // clear out others
10959                 cur.anim.stop(true);
10960             }
10961         }
10962         return this;
10963     },
10964
10965         /* @private */
10966     beforeFx : function(o){
10967         if(this.hasActiveFx() && !o.concurrent){
10968            if(o.stopFx){
10969                this.stopFx();
10970                return true;
10971            }
10972            return false;
10973         }
10974         return true;
10975     },
10976
10977         /**
10978          * Returns true if the element is currently blocking so that no other effect can be queued
10979          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10980          * used to ensure that an effect initiated by a user action runs to completion prior to the
10981          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10982          * @return {Boolean} True if blocking, else false
10983          */
10984     hasFxBlock : function(){
10985         var q = this.fxQueue;
10986         return q && q[0] && q[0].block;
10987     },
10988
10989         /* @private */
10990     queueFx : function(o, fn){
10991         if(!this.fxQueue){
10992             this.fxQueue = [];
10993         }
10994         if(!this.hasFxBlock()){
10995             Roo.applyIf(o, this.fxDefaults);
10996             if(!o.concurrent){
10997                 var run = this.beforeFx(o);
10998                 fn.block = o.block;
10999                 this.fxQueue.push(fn);
11000                 if(run){
11001                     this.nextFx();
11002                 }
11003             }else{
11004                 fn.call(this);
11005             }
11006         }
11007         return this;
11008     },
11009
11010         /* @private */
11011     fxWrap : function(pos, o, vis){
11012         var wrap;
11013         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11014             var wrapXY;
11015             if(o.fixPosition){
11016                 wrapXY = this.getXY();
11017             }
11018             var div = document.createElement("div");
11019             div.style.visibility = vis;
11020             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11021             wrap.setPositioning(pos);
11022             if(wrap.getStyle("position") == "static"){
11023                 wrap.position("relative");
11024             }
11025             this.clearPositioning('auto');
11026             wrap.clip();
11027             wrap.dom.appendChild(this.dom);
11028             if(wrapXY){
11029                 wrap.setXY(wrapXY);
11030             }
11031         }
11032         return wrap;
11033     },
11034
11035         /* @private */
11036     fxUnwrap : function(wrap, pos, o){
11037         this.clearPositioning();
11038         this.setPositioning(pos);
11039         if(!o.wrap){
11040             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11041             wrap.remove();
11042         }
11043     },
11044
11045         /* @private */
11046     getFxRestore : function(){
11047         var st = this.dom.style;
11048         return {pos: this.getPositioning(), width: st.width, height : st.height};
11049     },
11050
11051         /* @private */
11052     afterFx : function(o){
11053         if(o.afterStyle){
11054             this.applyStyles(o.afterStyle);
11055         }
11056         if(o.afterCls){
11057             this.addClass(o.afterCls);
11058         }
11059         if(o.remove === true){
11060             this.remove();
11061         }
11062         Roo.callback(o.callback, o.scope, [this]);
11063         if(!o.concurrent){
11064             this.fxQueue.shift();
11065             this.nextFx();
11066         }
11067     },
11068
11069         /* @private */
11070     getFxEl : function(){ // support for composite element fx
11071         return Roo.get(this.dom);
11072     },
11073
11074         /* @private */
11075     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11076         animType = animType || 'run';
11077         opt = opt || {};
11078         var anim = Roo.lib.Anim[animType](
11079             this.dom, args,
11080             (opt.duration || defaultDur) || .35,
11081             (opt.easing || defaultEase) || 'easeOut',
11082             function(){
11083                 Roo.callback(cb, this);
11084             },
11085             this
11086         );
11087         opt.anim = anim;
11088         return anim;
11089     }
11090 };
11091
11092 // backwords compat
11093 Roo.Fx.resize = Roo.Fx.scale;
11094
11095 //When included, Roo.Fx is automatically applied to Element so that all basic
11096 //effects are available directly via the Element API
11097 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11098  * Based on:
11099  * Ext JS Library 1.1.1
11100  * Copyright(c) 2006-2007, Ext JS, LLC.
11101  *
11102  * Originally Released Under LGPL - original licence link has changed is not relivant.
11103  *
11104  * Fork - LGPL
11105  * <script type="text/javascript">
11106  */
11107
11108
11109 /**
11110  * @class Roo.CompositeElement
11111  * Standard composite class. Creates a Roo.Element for every element in the collection.
11112  * <br><br>
11113  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11114  * actions will be performed on all the elements in this collection.</b>
11115  * <br><br>
11116  * All methods return <i>this</i> and can be chained.
11117  <pre><code>
11118  var els = Roo.select("#some-el div.some-class", true);
11119  // or select directly from an existing element
11120  var el = Roo.get('some-el');
11121  el.select('div.some-class', true);
11122
11123  els.setWidth(100); // all elements become 100 width
11124  els.hide(true); // all elements fade out and hide
11125  // or
11126  els.setWidth(100).hide(true);
11127  </code></pre>
11128  */
11129 Roo.CompositeElement = function(els){
11130     this.elements = [];
11131     this.addElements(els);
11132 };
11133 Roo.CompositeElement.prototype = {
11134     isComposite: true,
11135     addElements : function(els){
11136         if(!els) {
11137             return this;
11138         }
11139         if(typeof els == "string"){
11140             els = Roo.Element.selectorFunction(els);
11141         }
11142         var yels = this.elements;
11143         var index = yels.length-1;
11144         for(var i = 0, len = els.length; i < len; i++) {
11145                 yels[++index] = Roo.get(els[i]);
11146         }
11147         return this;
11148     },
11149
11150     /**
11151     * Clears this composite and adds the elements returned by the passed selector.
11152     * @param {String/Array} els A string CSS selector, an array of elements or an element
11153     * @return {CompositeElement} this
11154     */
11155     fill : function(els){
11156         this.elements = [];
11157         this.add(els);
11158         return this;
11159     },
11160
11161     /**
11162     * Filters this composite to only elements that match the passed selector.
11163     * @param {String} selector A string CSS selector
11164     * @param {Boolean} inverse return inverse filter (not matches)
11165     * @return {CompositeElement} this
11166     */
11167     filter : function(selector, inverse){
11168         var els = [];
11169         inverse = inverse || false;
11170         this.each(function(el){
11171             var match = inverse ? !el.is(selector) : el.is(selector);
11172             if(match){
11173                 els[els.length] = el.dom;
11174             }
11175         });
11176         this.fill(els);
11177         return this;
11178     },
11179
11180     invoke : function(fn, args){
11181         var els = this.elements;
11182         for(var i = 0, len = els.length; i < len; i++) {
11183                 Roo.Element.prototype[fn].apply(els[i], args);
11184         }
11185         return this;
11186     },
11187     /**
11188     * Adds elements to this composite.
11189     * @param {String/Array} els A string CSS selector, an array of elements or an element
11190     * @return {CompositeElement} this
11191     */
11192     add : function(els){
11193         if(typeof els == "string"){
11194             this.addElements(Roo.Element.selectorFunction(els));
11195         }else if(els.length !== undefined){
11196             this.addElements(els);
11197         }else{
11198             this.addElements([els]);
11199         }
11200         return this;
11201     },
11202     /**
11203     * Calls the passed function passing (el, this, index) for each element in this composite.
11204     * @param {Function} fn The function to call
11205     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11206     * @return {CompositeElement} this
11207     */
11208     each : function(fn, scope){
11209         var els = this.elements;
11210         for(var i = 0, len = els.length; i < len; i++){
11211             if(fn.call(scope || els[i], els[i], this, i) === false) {
11212                 break;
11213             }
11214         }
11215         return this;
11216     },
11217
11218     /**
11219      * Returns the Element object at the specified index
11220      * @param {Number} index
11221      * @return {Roo.Element}
11222      */
11223     item : function(index){
11224         return this.elements[index] || null;
11225     },
11226
11227     /**
11228      * Returns the first Element
11229      * @return {Roo.Element}
11230      */
11231     first : function(){
11232         return this.item(0);
11233     },
11234
11235     /**
11236      * Returns the last Element
11237      * @return {Roo.Element}
11238      */
11239     last : function(){
11240         return this.item(this.elements.length-1);
11241     },
11242
11243     /**
11244      * Returns the number of elements in this composite
11245      * @return Number
11246      */
11247     getCount : function(){
11248         return this.elements.length;
11249     },
11250
11251     /**
11252      * Returns true if this composite contains the passed element
11253      * @return Boolean
11254      */
11255     contains : function(el){
11256         return this.indexOf(el) !== -1;
11257     },
11258
11259     /**
11260      * Returns true if this composite contains the passed element
11261      * @return Boolean
11262      */
11263     indexOf : function(el){
11264         return this.elements.indexOf(Roo.get(el));
11265     },
11266
11267
11268     /**
11269     * Removes the specified element(s).
11270     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11271     * or an array of any of those.
11272     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11273     * @return {CompositeElement} this
11274     */
11275     removeElement : function(el, removeDom){
11276         if(el instanceof Array){
11277             for(var i = 0, len = el.length; i < len; i++){
11278                 this.removeElement(el[i]);
11279             }
11280             return this;
11281         }
11282         var index = typeof el == 'number' ? el : this.indexOf(el);
11283         if(index !== -1){
11284             if(removeDom){
11285                 var d = this.elements[index];
11286                 if(d.dom){
11287                     d.remove();
11288                 }else{
11289                     d.parentNode.removeChild(d);
11290                 }
11291             }
11292             this.elements.splice(index, 1);
11293         }
11294         return this;
11295     },
11296
11297     /**
11298     * Replaces the specified element with the passed element.
11299     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11300     * to replace.
11301     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11302     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11303     * @return {CompositeElement} this
11304     */
11305     replaceElement : function(el, replacement, domReplace){
11306         var index = typeof el == 'number' ? el : this.indexOf(el);
11307         if(index !== -1){
11308             if(domReplace){
11309                 this.elements[index].replaceWith(replacement);
11310             }else{
11311                 this.elements.splice(index, 1, Roo.get(replacement))
11312             }
11313         }
11314         return this;
11315     },
11316
11317     /**
11318      * Removes all elements.
11319      */
11320     clear : function(){
11321         this.elements = [];
11322     }
11323 };
11324 (function(){
11325     Roo.CompositeElement.createCall = function(proto, fnName){
11326         if(!proto[fnName]){
11327             proto[fnName] = function(){
11328                 return this.invoke(fnName, arguments);
11329             };
11330         }
11331     };
11332     for(var fnName in Roo.Element.prototype){
11333         if(typeof Roo.Element.prototype[fnName] == "function"){
11334             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11335         }
11336     };
11337 })();
11338 /*
11339  * Based on:
11340  * Ext JS Library 1.1.1
11341  * Copyright(c) 2006-2007, Ext JS, LLC.
11342  *
11343  * Originally Released Under LGPL - original licence link has changed is not relivant.
11344  *
11345  * Fork - LGPL
11346  * <script type="text/javascript">
11347  */
11348
11349 /**
11350  * @class Roo.CompositeElementLite
11351  * @extends Roo.CompositeElement
11352  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11353  <pre><code>
11354  var els = Roo.select("#some-el div.some-class");
11355  // or select directly from an existing element
11356  var el = Roo.get('some-el');
11357  el.select('div.some-class');
11358
11359  els.setWidth(100); // all elements become 100 width
11360  els.hide(true); // all elements fade out and hide
11361  // or
11362  els.setWidth(100).hide(true);
11363  </code></pre><br><br>
11364  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11365  * actions will be performed on all the elements in this collection.</b>
11366  */
11367 Roo.CompositeElementLite = function(els){
11368     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11369     this.el = new Roo.Element.Flyweight();
11370 };
11371 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11372     addElements : function(els){
11373         if(els){
11374             if(els instanceof Array){
11375                 this.elements = this.elements.concat(els);
11376             }else{
11377                 var yels = this.elements;
11378                 var index = yels.length-1;
11379                 for(var i = 0, len = els.length; i < len; i++) {
11380                     yels[++index] = els[i];
11381                 }
11382             }
11383         }
11384         return this;
11385     },
11386     invoke : function(fn, args){
11387         var els = this.elements;
11388         var el = this.el;
11389         for(var i = 0, len = els.length; i < len; i++) {
11390             el.dom = els[i];
11391                 Roo.Element.prototype[fn].apply(el, args);
11392         }
11393         return this;
11394     },
11395     /**
11396      * Returns a flyweight Element of the dom element object at the specified index
11397      * @param {Number} index
11398      * @return {Roo.Element}
11399      */
11400     item : function(index){
11401         if(!this.elements[index]){
11402             return null;
11403         }
11404         this.el.dom = this.elements[index];
11405         return this.el;
11406     },
11407
11408     // fixes scope with flyweight
11409     addListener : function(eventName, handler, scope, opt){
11410         var els = this.elements;
11411         for(var i = 0, len = els.length; i < len; i++) {
11412             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11413         }
11414         return this;
11415     },
11416
11417     /**
11418     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11419     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11420     * a reference to the dom node, use el.dom.</b>
11421     * @param {Function} fn The function to call
11422     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11423     * @return {CompositeElement} this
11424     */
11425     each : function(fn, scope){
11426         var els = this.elements;
11427         var el = this.el;
11428         for(var i = 0, len = els.length; i < len; i++){
11429             el.dom = els[i];
11430                 if(fn.call(scope || el, el, this, i) === false){
11431                 break;
11432             }
11433         }
11434         return this;
11435     },
11436
11437     indexOf : function(el){
11438         return this.elements.indexOf(Roo.getDom(el));
11439     },
11440
11441     replaceElement : function(el, replacement, domReplace){
11442         var index = typeof el == 'number' ? el : this.indexOf(el);
11443         if(index !== -1){
11444             replacement = Roo.getDom(replacement);
11445             if(domReplace){
11446                 var d = this.elements[index];
11447                 d.parentNode.insertBefore(replacement, d);
11448                 d.parentNode.removeChild(d);
11449             }
11450             this.elements.splice(index, 1, replacement);
11451         }
11452         return this;
11453     }
11454 });
11455 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11456
11457 /*
11458  * Based on:
11459  * Ext JS Library 1.1.1
11460  * Copyright(c) 2006-2007, Ext JS, LLC.
11461  *
11462  * Originally Released Under LGPL - original licence link has changed is not relivant.
11463  *
11464  * Fork - LGPL
11465  * <script type="text/javascript">
11466  */
11467
11468  
11469
11470 /**
11471  * @class Roo.data.Connection
11472  * @extends Roo.util.Observable
11473  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11474  * either to a configured URL, or to a URL specified at request time. 
11475  * 
11476  * Requests made by this class are asynchronous, and will return immediately. No data from
11477  * the server will be available to the statement immediately following the {@link #request} call.
11478  * To process returned data, use a callback in the request options object, or an event listener.
11479  * 
11480  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11481  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11482  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11483  * property and, if present, the IFRAME's XML document as the responseXML property.
11484  * 
11485  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11486  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11487  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11488  * standard DOM methods.
11489  * @constructor
11490  * @param {Object} config a configuration object.
11491  */
11492 Roo.data.Connection = function(config){
11493     Roo.apply(this, config);
11494     this.addEvents({
11495         /**
11496          * @event beforerequest
11497          * Fires before a network request is made to retrieve a data object.
11498          * @param {Connection} conn This Connection object.
11499          * @param {Object} options The options config object passed to the {@link #request} method.
11500          */
11501         "beforerequest" : true,
11502         /**
11503          * @event requestcomplete
11504          * Fires if the request was successfully completed.
11505          * @param {Connection} conn This Connection object.
11506          * @param {Object} response The XHR object containing the response data.
11507          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11508          * @param {Object} options The options config object passed to the {@link #request} method.
11509          */
11510         "requestcomplete" : true,
11511         /**
11512          * @event requestexception
11513          * Fires if an error HTTP status was returned from the server.
11514          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11515          * @param {Connection} conn This Connection object.
11516          * @param {Object} response The XHR object containing the response data.
11517          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11518          * @param {Object} options The options config object passed to the {@link #request} method.
11519          */
11520         "requestexception" : true
11521     });
11522     Roo.data.Connection.superclass.constructor.call(this);
11523 };
11524
11525 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11526     /**
11527      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11528      */
11529     /**
11530      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11531      * extra parameters to each request made by this object. (defaults to undefined)
11532      */
11533     /**
11534      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11535      *  to each request made by this object. (defaults to undefined)
11536      */
11537     /**
11538      * @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)
11539      */
11540     /**
11541      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11542      */
11543     timeout : 30000,
11544     /**
11545      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11546      * @type Boolean
11547      */
11548     autoAbort:false,
11549
11550     /**
11551      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11552      * @type Boolean
11553      */
11554     disableCaching: true,
11555
11556     /**
11557      * Sends an HTTP request to a remote server.
11558      * @param {Object} options An object which may contain the following properties:<ul>
11559      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11560      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11561      * request, a url encoded string or a function to call to get either.</li>
11562      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11563      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11564      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11565      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11566      * <li>options {Object} The parameter to the request call.</li>
11567      * <li>success {Boolean} True if the request succeeded.</li>
11568      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11569      * </ul></li>
11570      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11571      * The callback is passed the following parameters:<ul>
11572      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11573      * <li>options {Object} The parameter to the request call.</li>
11574      * </ul></li>
11575      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11576      * The callback is passed the following parameters:<ul>
11577      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11578      * <li>options {Object} The parameter to the request call.</li>
11579      * </ul></li>
11580      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11581      * for the callback function. Defaults to the browser window.</li>
11582      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11583      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11584      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11585      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11586      * params for the post data. Any params will be appended to the URL.</li>
11587      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11588      * </ul>
11589      * @return {Number} transactionId
11590      */
11591     request : function(o){
11592         if(this.fireEvent("beforerequest", this, o) !== false){
11593             var p = o.params;
11594
11595             if(typeof p == "function"){
11596                 p = p.call(o.scope||window, o);
11597             }
11598             if(typeof p == "object"){
11599                 p = Roo.urlEncode(o.params);
11600             }
11601             if(this.extraParams){
11602                 var extras = Roo.urlEncode(this.extraParams);
11603                 p = p ? (p + '&' + extras) : extras;
11604             }
11605
11606             var url = o.url || this.url;
11607             if(typeof url == 'function'){
11608                 url = url.call(o.scope||window, o);
11609             }
11610
11611             if(o.form){
11612                 var form = Roo.getDom(o.form);
11613                 url = url || form.action;
11614
11615                 var enctype = form.getAttribute("enctype");
11616                 
11617                 if (o.formData) {
11618                     return this.doFormDataUpload(o,p,url);
11619                 }
11620                 
11621                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11622                     return this.doFormUpload(o, p, url);
11623                 }
11624                 var f = Roo.lib.Ajax.serializeForm(form);
11625                 p = p ? (p + '&' + f) : f;
11626             }
11627
11628             var hs = o.headers;
11629             if(this.defaultHeaders){
11630                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11631                 if(!o.headers){
11632                     o.headers = hs;
11633                 }
11634             }
11635
11636             var cb = {
11637                 success: this.handleResponse,
11638                 failure: this.handleFailure,
11639                 scope: this,
11640                 argument: {options: o},
11641                 timeout : o.timeout || this.timeout
11642             };
11643
11644             var method = o.method||this.method||(p ? "POST" : "GET");
11645
11646             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11647                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11648             }
11649
11650             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11651                 if(o.autoAbort){
11652                     this.abort();
11653                 }
11654             }else if(this.autoAbort !== false){
11655                 this.abort();
11656             }
11657
11658             if((method == 'GET' && p) || o.xmlData){
11659                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11660                 p = '';
11661             }
11662             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11663             return this.transId;
11664         }else{
11665             Roo.callback(o.callback, o.scope, [o, null, null]);
11666             return null;
11667         }
11668     },
11669
11670     /**
11671      * Determine whether this object has a request outstanding.
11672      * @param {Number} transactionId (Optional) defaults to the last transaction
11673      * @return {Boolean} True if there is an outstanding request.
11674      */
11675     isLoading : function(transId){
11676         if(transId){
11677             return Roo.lib.Ajax.isCallInProgress(transId);
11678         }else{
11679             return this.transId ? true : false;
11680         }
11681     },
11682
11683     /**
11684      * Aborts any outstanding request.
11685      * @param {Number} transactionId (Optional) defaults to the last transaction
11686      */
11687     abort : function(transId){
11688         if(transId || this.isLoading()){
11689             Roo.lib.Ajax.abort(transId || this.transId);
11690         }
11691     },
11692
11693     // private
11694     handleResponse : function(response){
11695         this.transId = false;
11696         var options = response.argument.options;
11697         response.argument = options ? options.argument : null;
11698         this.fireEvent("requestcomplete", this, response, options);
11699         Roo.callback(options.success, options.scope, [response, options]);
11700         Roo.callback(options.callback, options.scope, [options, true, response]);
11701     },
11702
11703     // private
11704     handleFailure : function(response, e){
11705         this.transId = false;
11706         var options = response.argument.options;
11707         response.argument = options ? options.argument : null;
11708         this.fireEvent("requestexception", this, response, options, e);
11709         Roo.callback(options.failure, options.scope, [response, options]);
11710         Roo.callback(options.callback, options.scope, [options, false, response]);
11711     },
11712
11713     // private
11714     doFormUpload : function(o, ps, url){
11715         var id = Roo.id();
11716         var frame = document.createElement('iframe');
11717         frame.id = id;
11718         frame.name = id;
11719         frame.className = 'x-hidden';
11720         if(Roo.isIE){
11721             frame.src = Roo.SSL_SECURE_URL;
11722         }
11723         document.body.appendChild(frame);
11724
11725         if(Roo.isIE){
11726            document.frames[id].name = id;
11727         }
11728
11729         var form = Roo.getDom(o.form);
11730         form.target = id;
11731         form.method = 'POST';
11732         form.enctype = form.encoding = 'multipart/form-data';
11733         if(url){
11734             form.action = url;
11735         }
11736
11737         var hiddens, hd;
11738         if(ps){ // add dynamic params
11739             hiddens = [];
11740             ps = Roo.urlDecode(ps, false);
11741             for(var k in ps){
11742                 if(ps.hasOwnProperty(k)){
11743                     hd = document.createElement('input');
11744                     hd.type = 'hidden';
11745                     hd.name = k;
11746                     hd.value = ps[k];
11747                     form.appendChild(hd);
11748                     hiddens.push(hd);
11749                 }
11750             }
11751         }
11752
11753         function cb(){
11754             var r = {  // bogus response object
11755                 responseText : '',
11756                 responseXML : null
11757             };
11758
11759             r.argument = o ? o.argument : null;
11760
11761             try { //
11762                 var doc;
11763                 if(Roo.isIE){
11764                     doc = frame.contentWindow.document;
11765                 }else {
11766                     doc = (frame.contentDocument || window.frames[id].document);
11767                 }
11768                 if(doc && doc.body){
11769                     r.responseText = doc.body.innerHTML;
11770                 }
11771                 if(doc && doc.XMLDocument){
11772                     r.responseXML = doc.XMLDocument;
11773                 }else {
11774                     r.responseXML = doc;
11775                 }
11776             }
11777             catch(e) {
11778                 // ignore
11779             }
11780
11781             Roo.EventManager.removeListener(frame, 'load', cb, this);
11782
11783             this.fireEvent("requestcomplete", this, r, o);
11784             Roo.callback(o.success, o.scope, [r, o]);
11785             Roo.callback(o.callback, o.scope, [o, true, r]);
11786
11787             setTimeout(function(){document.body.removeChild(frame);}, 100);
11788         }
11789
11790         Roo.EventManager.on(frame, 'load', cb, this);
11791         form.submit();
11792
11793         if(hiddens){ // remove dynamic params
11794             for(var i = 0, len = hiddens.length; i < len; i++){
11795                 form.removeChild(hiddens[i]);
11796             }
11797         }
11798     },
11799     // this is a 'formdata version???'
11800     
11801     
11802     doFormDataUpload : function(o, ps, url)
11803     {
11804         var form = Roo.getDom(o.form);
11805         form.enctype = form.encoding = 'multipart/form-data';
11806         var formData = o.formData === true ? new FormData(form) : o.formData;
11807       
11808         var cb = {
11809             success: this.handleResponse,
11810             failure: this.handleFailure,
11811             scope: this,
11812             argument: {options: o},
11813             timeout : o.timeout || this.timeout
11814         };
11815  
11816         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11817             if(o.autoAbort){
11818                 this.abort();
11819             }
11820         }else if(this.autoAbort !== false){
11821             this.abort();
11822         }
11823
11824         //Roo.lib.Ajax.defaultPostHeader = null;
11825         Roo.lib.Ajax.useDefaultHeader = false;
11826         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11827         Roo.lib.Ajax.useDefaultHeader = true;
11828  
11829          
11830     }
11831     
11832 });
11833 /*
11834  * Based on:
11835  * Ext JS Library 1.1.1
11836  * Copyright(c) 2006-2007, Ext JS, LLC.
11837  *
11838  * Originally Released Under LGPL - original licence link has changed is not relivant.
11839  *
11840  * Fork - LGPL
11841  * <script type="text/javascript">
11842  */
11843  
11844 /**
11845  * Global Ajax request class.
11846  * 
11847  * @class Roo.Ajax
11848  * @extends Roo.data.Connection
11849  * @static
11850  * 
11851  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11852  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11853  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11854  * @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)
11855  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11856  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11857  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11858  */
11859 Roo.Ajax = new Roo.data.Connection({
11860     // fix up the docs
11861     /**
11862      * @scope Roo.Ajax
11863      * @type {Boolear} 
11864      */
11865     autoAbort : false,
11866
11867     /**
11868      * Serialize the passed form into a url encoded string
11869      * @scope Roo.Ajax
11870      * @param {String/HTMLElement} form
11871      * @return {String}
11872      */
11873     serializeForm : function(form){
11874         return Roo.lib.Ajax.serializeForm(form);
11875     }
11876 });/*
11877  * Based on:
11878  * Ext JS Library 1.1.1
11879  * Copyright(c) 2006-2007, Ext JS, LLC.
11880  *
11881  * Originally Released Under LGPL - original licence link has changed is not relivant.
11882  *
11883  * Fork - LGPL
11884  * <script type="text/javascript">
11885  */
11886
11887  
11888 /**
11889  * @class Roo.UpdateManager
11890  * @extends Roo.util.Observable
11891  * Provides AJAX-style update for Element object.<br><br>
11892  * Usage:<br>
11893  * <pre><code>
11894  * // Get it from a Roo.Element object
11895  * var el = Roo.get("foo");
11896  * var mgr = el.getUpdateManager();
11897  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11898  * ...
11899  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11900  * <br>
11901  * // or directly (returns the same UpdateManager instance)
11902  * var mgr = new Roo.UpdateManager("myElementId");
11903  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11904  * mgr.on("update", myFcnNeedsToKnow);
11905  * <br>
11906    // short handed call directly from the element object
11907    Roo.get("foo").load({
11908         url: "bar.php",
11909         scripts:true,
11910         params: "for=bar",
11911         text: "Loading Foo..."
11912    });
11913  * </code></pre>
11914  * @constructor
11915  * Create new UpdateManager directly.
11916  * @param {String/HTMLElement/Roo.Element} el The element to update
11917  * @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).
11918  */
11919 Roo.UpdateManager = function(el, forceNew){
11920     el = Roo.get(el);
11921     if(!forceNew && el.updateManager){
11922         return el.updateManager;
11923     }
11924     /**
11925      * The Element object
11926      * @type Roo.Element
11927      */
11928     this.el = el;
11929     /**
11930      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11931      * @type String
11932      */
11933     this.defaultUrl = null;
11934
11935     this.addEvents({
11936         /**
11937          * @event beforeupdate
11938          * Fired before an update is made, return false from your handler and the update is cancelled.
11939          * @param {Roo.Element} el
11940          * @param {String/Object/Function} url
11941          * @param {String/Object} params
11942          */
11943         "beforeupdate": true,
11944         /**
11945          * @event update
11946          * Fired after successful update is made.
11947          * @param {Roo.Element} el
11948          * @param {Object} oResponseObject The response Object
11949          */
11950         "update": true,
11951         /**
11952          * @event failure
11953          * Fired on update failure.
11954          * @param {Roo.Element} el
11955          * @param {Object} oResponseObject The response Object
11956          */
11957         "failure": true
11958     });
11959     var d = Roo.UpdateManager.defaults;
11960     /**
11961      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11962      * @type String
11963      */
11964     this.sslBlankUrl = d.sslBlankUrl;
11965     /**
11966      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11967      * @type Boolean
11968      */
11969     this.disableCaching = d.disableCaching;
11970     /**
11971      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11972      * @type String
11973      */
11974     this.indicatorText = d.indicatorText;
11975     /**
11976      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11977      * @type String
11978      */
11979     this.showLoadIndicator = d.showLoadIndicator;
11980     /**
11981      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11982      * @type Number
11983      */
11984     this.timeout = d.timeout;
11985
11986     /**
11987      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11988      * @type Boolean
11989      */
11990     this.loadScripts = d.loadScripts;
11991
11992     /**
11993      * Transaction object of current executing transaction
11994      */
11995     this.transaction = null;
11996
11997     /**
11998      * @private
11999      */
12000     this.autoRefreshProcId = null;
12001     /**
12002      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12003      * @type Function
12004      */
12005     this.refreshDelegate = this.refresh.createDelegate(this);
12006     /**
12007      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.updateDelegate = this.update.createDelegate(this);
12011     /**
12012      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12016     /**
12017      * @private
12018      */
12019     this.successDelegate = this.processSuccess.createDelegate(this);
12020     /**
12021      * @private
12022      */
12023     this.failureDelegate = this.processFailure.createDelegate(this);
12024
12025     if(!this.renderer){
12026      /**
12027       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12028       */
12029     this.renderer = new Roo.UpdateManager.BasicRenderer();
12030     }
12031     
12032     Roo.UpdateManager.superclass.constructor.call(this);
12033 };
12034
12035 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12036     /**
12037      * Get the Element this UpdateManager is bound to
12038      * @return {Roo.Element} The element
12039      */
12040     getEl : function(){
12041         return this.el;
12042     },
12043     /**
12044      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12045      * @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:
12046 <pre><code>
12047 um.update({<br/>
12048     url: "your-url.php",<br/>
12049     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12050     callback: yourFunction,<br/>
12051     scope: yourObject, //(optional scope)  <br/>
12052     discardUrl: false, <br/>
12053     nocache: false,<br/>
12054     text: "Loading...",<br/>
12055     timeout: 30,<br/>
12056     scripts: false<br/>
12057 });
12058 </code></pre>
12059      * The only required property is url. The optional properties nocache, text and scripts
12060      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12061      * @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}
12062      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12063      * @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.
12064      */
12065     update : function(url, params, callback, discardUrl){
12066         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12067             var method = this.method,
12068                 cfg;
12069             if(typeof url == "object"){ // must be config object
12070                 cfg = url;
12071                 url = cfg.url;
12072                 params = params || cfg.params;
12073                 callback = callback || cfg.callback;
12074                 discardUrl = discardUrl || cfg.discardUrl;
12075                 if(callback && cfg.scope){
12076                     callback = callback.createDelegate(cfg.scope);
12077                 }
12078                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12079                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12080                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12081                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12082                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12083             }
12084             this.showLoading();
12085             if(!discardUrl){
12086                 this.defaultUrl = url;
12087             }
12088             if(typeof url == "function"){
12089                 url = url.call(this);
12090             }
12091
12092             method = method || (params ? "POST" : "GET");
12093             if(method == "GET"){
12094                 url = this.prepareUrl(url);
12095             }
12096
12097             var o = Roo.apply(cfg ||{}, {
12098                 url : url,
12099                 params: params,
12100                 success: this.successDelegate,
12101                 failure: this.failureDelegate,
12102                 callback: undefined,
12103                 timeout: (this.timeout*1000),
12104                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12105             });
12106             Roo.log("updated manager called with timeout of " + o.timeout);
12107             this.transaction = Roo.Ajax.request(o);
12108         }
12109     },
12110
12111     /**
12112      * 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.
12113      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12114      * @param {String/HTMLElement} form The form Id or form element
12115      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12116      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12117      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12118      */
12119     formUpdate : function(form, url, reset, callback){
12120         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12121             if(typeof url == "function"){
12122                 url = url.call(this);
12123             }
12124             form = Roo.getDom(form);
12125             this.transaction = Roo.Ajax.request({
12126                 form: form,
12127                 url:url,
12128                 success: this.successDelegate,
12129                 failure: this.failureDelegate,
12130                 timeout: (this.timeout*1000),
12131                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12132             });
12133             this.showLoading.defer(1, this);
12134         }
12135     },
12136
12137     /**
12138      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12139      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12140      */
12141     refresh : function(callback){
12142         if(this.defaultUrl == null){
12143             return;
12144         }
12145         this.update(this.defaultUrl, null, callback, true);
12146     },
12147
12148     /**
12149      * Set this element to auto refresh.
12150      * @param {Number} interval How often to update (in seconds).
12151      * @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)
12152      * @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}
12153      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12154      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12155      */
12156     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12157         if(refreshNow){
12158             this.update(url || this.defaultUrl, params, callback, true);
12159         }
12160         if(this.autoRefreshProcId){
12161             clearInterval(this.autoRefreshProcId);
12162         }
12163         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12164     },
12165
12166     /**
12167      * Stop auto refresh on this element.
12168      */
12169      stopAutoRefresh : function(){
12170         if(this.autoRefreshProcId){
12171             clearInterval(this.autoRefreshProcId);
12172             delete this.autoRefreshProcId;
12173         }
12174     },
12175
12176     isAutoRefreshing : function(){
12177        return this.autoRefreshProcId ? true : false;
12178     },
12179     /**
12180      * Called to update the element to "Loading" state. Override to perform custom action.
12181      */
12182     showLoading : function(){
12183         if(this.showLoadIndicator){
12184             this.el.update(this.indicatorText);
12185         }
12186     },
12187
12188     /**
12189      * Adds unique parameter to query string if disableCaching = true
12190      * @private
12191      */
12192     prepareUrl : function(url){
12193         if(this.disableCaching){
12194             var append = "_dc=" + (new Date().getTime());
12195             if(url.indexOf("?") !== -1){
12196                 url += "&" + append;
12197             }else{
12198                 url += "?" + append;
12199             }
12200         }
12201         return url;
12202     },
12203
12204     /**
12205      * @private
12206      */
12207     processSuccess : function(response){
12208         this.transaction = null;
12209         if(response.argument.form && response.argument.reset){
12210             try{ // put in try/catch since some older FF releases had problems with this
12211                 response.argument.form.reset();
12212             }catch(e){}
12213         }
12214         if(this.loadScripts){
12215             this.renderer.render(this.el, response, this,
12216                 this.updateComplete.createDelegate(this, [response]));
12217         }else{
12218             this.renderer.render(this.el, response, this);
12219             this.updateComplete(response);
12220         }
12221     },
12222
12223     updateComplete : function(response){
12224         this.fireEvent("update", this.el, response);
12225         if(typeof response.argument.callback == "function"){
12226             response.argument.callback(this.el, true, response);
12227         }
12228     },
12229
12230     /**
12231      * @private
12232      */
12233     processFailure : function(response){
12234         this.transaction = null;
12235         this.fireEvent("failure", this.el, response);
12236         if(typeof response.argument.callback == "function"){
12237             response.argument.callback(this.el, false, response);
12238         }
12239     },
12240
12241     /**
12242      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12243      * @param {Object} renderer The object implementing the render() method
12244      */
12245     setRenderer : function(renderer){
12246         this.renderer = renderer;
12247     },
12248
12249     getRenderer : function(){
12250        return this.renderer;
12251     },
12252
12253     /**
12254      * Set the defaultUrl used for updates
12255      * @param {String/Function} defaultUrl The url or a function to call to get the url
12256      */
12257     setDefaultUrl : function(defaultUrl){
12258         this.defaultUrl = defaultUrl;
12259     },
12260
12261     /**
12262      * Aborts the executing transaction
12263      */
12264     abort : function(){
12265         if(this.transaction){
12266             Roo.Ajax.abort(this.transaction);
12267         }
12268     },
12269
12270     /**
12271      * Returns true if an update is in progress
12272      * @return {Boolean}
12273      */
12274     isUpdating : function(){
12275         if(this.transaction){
12276             return Roo.Ajax.isLoading(this.transaction);
12277         }
12278         return false;
12279     }
12280 });
12281
12282 /**
12283  * @class Roo.UpdateManager.defaults
12284  * @static (not really - but it helps the doc tool)
12285  * The defaults collection enables customizing the default properties of UpdateManager
12286  */
12287    Roo.UpdateManager.defaults = {
12288        /**
12289          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12290          * @type Number
12291          */
12292          timeout : 30,
12293
12294          /**
12295          * True to process scripts by default (Defaults to false).
12296          * @type Boolean
12297          */
12298         loadScripts : false,
12299
12300         /**
12301         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12302         * @type String
12303         */
12304         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12305         /**
12306          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12307          * @type Boolean
12308          */
12309         disableCaching : false,
12310         /**
12311          * Whether to show indicatorText when loading (Defaults to true).
12312          * @type Boolean
12313          */
12314         showLoadIndicator : true,
12315         /**
12316          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12317          * @type String
12318          */
12319         indicatorText : '<div class="loading-indicator">Loading...</div>'
12320    };
12321
12322 /**
12323  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12324  *Usage:
12325  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12326  * @param {String/HTMLElement/Roo.Element} el The element to update
12327  * @param {String} url The url
12328  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12329  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12330  * @static
12331  * @deprecated
12332  * @member Roo.UpdateManager
12333  */
12334 Roo.UpdateManager.updateElement = function(el, url, params, options){
12335     var um = Roo.get(el, true).getUpdateManager();
12336     Roo.apply(um, options);
12337     um.update(url, params, options ? options.callback : null);
12338 };
12339 // alias for backwards compat
12340 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12341 /**
12342  * @class Roo.UpdateManager.BasicRenderer
12343  * Default Content renderer. Updates the elements innerHTML with the responseText.
12344  */
12345 Roo.UpdateManager.BasicRenderer = function(){};
12346
12347 Roo.UpdateManager.BasicRenderer.prototype = {
12348     /**
12349      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12350      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12351      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12352      * @param {Roo.Element} el The element being rendered
12353      * @param {Object} response The YUI Connect response object
12354      * @param {UpdateManager} updateManager The calling update manager
12355      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12356      */
12357      render : function(el, response, updateManager, callback){
12358         el.update(response.responseText, updateManager.loadScripts, callback);
12359     }
12360 };
12361 /*
12362  * Based on:
12363  * Roo JS
12364  * (c)) Alan Knowles
12365  * Licence : LGPL
12366  */
12367
12368
12369 /**
12370  * @class Roo.DomTemplate
12371  * @extends Roo.Template
12372  * An effort at a dom based template engine..
12373  *
12374  * Similar to XTemplate, except it uses dom parsing to create the template..
12375  *
12376  * Supported features:
12377  *
12378  *  Tags:
12379
12380 <pre><code>
12381       {a_variable} - output encoded.
12382       {a_variable.format:("Y-m-d")} - call a method on the variable
12383       {a_variable:raw} - unencoded output
12384       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12385       {a_variable:this.method_on_template(...)} - call a method on the template object.
12386  
12387 </code></pre>
12388  *  The tpl tag:
12389 <pre><code>
12390         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12391         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12392         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12393         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12394   
12395 </code></pre>
12396  *      
12397  */
12398 Roo.DomTemplate = function()
12399 {
12400      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12401      if (this.html) {
12402         this.compile();
12403      }
12404 };
12405
12406
12407 Roo.extend(Roo.DomTemplate, Roo.Template, {
12408     /**
12409      * id counter for sub templates.
12410      */
12411     id : 0,
12412     /**
12413      * flag to indicate if dom parser is inside a pre,
12414      * it will strip whitespace if not.
12415      */
12416     inPre : false,
12417     
12418     /**
12419      * The various sub templates
12420      */
12421     tpls : false,
12422     
12423     
12424     
12425     /**
12426      *
12427      * basic tag replacing syntax
12428      * WORD:WORD()
12429      *
12430      * // you can fake an object call by doing this
12431      *  x.t:(test,tesT) 
12432      * 
12433      */
12434     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12435     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12436     
12437     iterChild : function (node, method) {
12438         
12439         var oldPre = this.inPre;
12440         if (node.tagName == 'PRE') {
12441             this.inPre = true;
12442         }
12443         for( var i = 0; i < node.childNodes.length; i++) {
12444             method.call(this, node.childNodes[i]);
12445         }
12446         this.inPre = oldPre;
12447     },
12448     
12449     
12450     
12451     /**
12452      * compile the template
12453      *
12454      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12455      *
12456      */
12457     compile: function()
12458     {
12459         var s = this.html;
12460         
12461         // covert the html into DOM...
12462         var doc = false;
12463         var div =false;
12464         try {
12465             doc = document.implementation.createHTMLDocument("");
12466             doc.documentElement.innerHTML =   this.html  ;
12467             div = doc.documentElement;
12468         } catch (e) {
12469             // old IE... - nasty -- it causes all sorts of issues.. with
12470             // images getting pulled from server..
12471             div = document.createElement('div');
12472             div.innerHTML = this.html;
12473         }
12474         //doc.documentElement.innerHTML = htmlBody
12475          
12476         
12477         
12478         this.tpls = [];
12479         var _t = this;
12480         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12481         
12482         var tpls = this.tpls;
12483         
12484         // create a top level template from the snippet..
12485         
12486         //Roo.log(div.innerHTML);
12487         
12488         var tpl = {
12489             uid : 'master',
12490             id : this.id++,
12491             attr : false,
12492             value : false,
12493             body : div.innerHTML,
12494             
12495             forCall : false,
12496             execCall : false,
12497             dom : div,
12498             isTop : true
12499             
12500         };
12501         tpls.unshift(tpl);
12502         
12503         
12504         // compile them...
12505         this.tpls = [];
12506         Roo.each(tpls, function(tp){
12507             this.compileTpl(tp);
12508             this.tpls[tp.id] = tp;
12509         }, this);
12510         
12511         this.master = tpls[0];
12512         return this;
12513         
12514         
12515     },
12516     
12517     compileNode : function(node, istop) {
12518         // test for
12519         //Roo.log(node);
12520         
12521         
12522         // skip anything not a tag..
12523         if (node.nodeType != 1) {
12524             if (node.nodeType == 3 && !this.inPre) {
12525                 // reduce white space..
12526                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12527                 
12528             }
12529             return;
12530         }
12531         
12532         var tpl = {
12533             uid : false,
12534             id : false,
12535             attr : false,
12536             value : false,
12537             body : '',
12538             
12539             forCall : false,
12540             execCall : false,
12541             dom : false,
12542             isTop : istop
12543             
12544             
12545         };
12546         
12547         
12548         switch(true) {
12549             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12550             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12551             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12552             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12553             // no default..
12554         }
12555         
12556         
12557         if (!tpl.attr) {
12558             // just itterate children..
12559             this.iterChild(node,this.compileNode);
12560             return;
12561         }
12562         tpl.uid = this.id++;
12563         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12564         node.removeAttribute('roo-'+ tpl.attr);
12565         if (tpl.attr != 'name') {
12566             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12567             node.parentNode.replaceChild(placeholder,  node);
12568         } else {
12569             
12570             var placeholder =  document.createElement('span');
12571             placeholder.className = 'roo-tpl-' + tpl.value;
12572             node.parentNode.replaceChild(placeholder,  node);
12573         }
12574         
12575         // parent now sees '{domtplXXXX}
12576         this.iterChild(node,this.compileNode);
12577         
12578         // we should now have node body...
12579         var div = document.createElement('div');
12580         div.appendChild(node);
12581         tpl.dom = node;
12582         // this has the unfortunate side effect of converting tagged attributes
12583         // eg. href="{...}" into %7C...%7D
12584         // this has been fixed by searching for those combo's although it's a bit hacky..
12585         
12586         
12587         tpl.body = div.innerHTML;
12588         
12589         
12590          
12591         tpl.id = tpl.uid;
12592         switch(tpl.attr) {
12593             case 'for' :
12594                 switch (tpl.value) {
12595                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12596                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12597                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12598                 }
12599                 break;
12600             
12601             case 'exec':
12602                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12603                 break;
12604             
12605             case 'if':     
12606                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12607                 break;
12608             
12609             case 'name':
12610                 tpl.id  = tpl.value; // replace non characters???
12611                 break;
12612             
12613         }
12614         
12615         
12616         this.tpls.push(tpl);
12617         
12618         
12619         
12620     },
12621     
12622     
12623     
12624     
12625     /**
12626      * Compile a segment of the template into a 'sub-template'
12627      *
12628      * 
12629      * 
12630      *
12631      */
12632     compileTpl : function(tpl)
12633     {
12634         var fm = Roo.util.Format;
12635         var useF = this.disableFormats !== true;
12636         
12637         var sep = Roo.isGecko ? "+\n" : ",\n";
12638         
12639         var undef = function(str) {
12640             Roo.debug && Roo.log("Property not found :"  + str);
12641             return '';
12642         };
12643           
12644         //Roo.log(tpl.body);
12645         
12646         
12647         
12648         var fn = function(m, lbrace, name, format, args)
12649         {
12650             //Roo.log("ARGS");
12651             //Roo.log(arguments);
12652             args = args ? args.replace(/\\'/g,"'") : args;
12653             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12654             if (typeof(format) == 'undefined') {
12655                 format =  'htmlEncode'; 
12656             }
12657             if (format == 'raw' ) {
12658                 format = false;
12659             }
12660             
12661             if(name.substr(0, 6) == 'domtpl'){
12662                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12663             }
12664             
12665             // build an array of options to determine if value is undefined..
12666             
12667             // basically get 'xxxx.yyyy' then do
12668             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12669             //    (function () { Roo.log("Property not found"); return ''; })() :
12670             //    ......
12671             
12672             var udef_ar = [];
12673             var lookfor = '';
12674             Roo.each(name.split('.'), function(st) {
12675                 lookfor += (lookfor.length ? '.': '') + st;
12676                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12677             });
12678             
12679             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12680             
12681             
12682             if(format && useF){
12683                 
12684                 args = args ? ',' + args : "";
12685                  
12686                 if(format.substr(0, 5) != "this."){
12687                     format = "fm." + format + '(';
12688                 }else{
12689                     format = 'this.call("'+ format.substr(5) + '", ';
12690                     args = ", values";
12691                 }
12692                 
12693                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12694             }
12695              
12696             if (args && args.length) {
12697                 // called with xxyx.yuu:(test,test)
12698                 // change to ()
12699                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12700             }
12701             // raw.. - :raw modifier..
12702             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12703             
12704         };
12705         var body;
12706         // branched to use + in gecko and [].join() in others
12707         if(Roo.isGecko){
12708             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12709                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12710                     "';};};";
12711         }else{
12712             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12713             body.push(tpl.body.replace(/(\r\n|\n)/g,
12714                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12715             body.push("'].join('');};};");
12716             body = body.join('');
12717         }
12718         
12719         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12720        
12721         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12722         eval(body);
12723         
12724         return this;
12725     },
12726      
12727     /**
12728      * same as applyTemplate, except it's done to one of the subTemplates
12729      * when using named templates, you can do:
12730      *
12731      * var str = pl.applySubTemplate('your-name', values);
12732      *
12733      * 
12734      * @param {Number} id of the template
12735      * @param {Object} values to apply to template
12736      * @param {Object} parent (normaly the instance of this object)
12737      */
12738     applySubTemplate : function(id, values, parent)
12739     {
12740         
12741         
12742         var t = this.tpls[id];
12743         
12744         
12745         try { 
12746             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12747                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12748                 return '';
12749             }
12750         } catch(e) {
12751             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12752             Roo.log(values);
12753           
12754             return '';
12755         }
12756         try { 
12757             
12758             if(t.execCall && t.execCall.call(this, values, parent)){
12759                 return '';
12760             }
12761         } catch(e) {
12762             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12763             Roo.log(values);
12764             return '';
12765         }
12766         
12767         try {
12768             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12769             parent = t.target ? values : parent;
12770             if(t.forCall && vs instanceof Array){
12771                 var buf = [];
12772                 for(var i = 0, len = vs.length; i < len; i++){
12773                     try {
12774                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12775                     } catch (e) {
12776                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12777                         Roo.log(e.body);
12778                         //Roo.log(t.compiled);
12779                         Roo.log(vs[i]);
12780                     }   
12781                 }
12782                 return buf.join('');
12783             }
12784         } catch (e) {
12785             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12786             Roo.log(values);
12787             return '';
12788         }
12789         try {
12790             return t.compiled.call(this, vs, parent);
12791         } catch (e) {
12792             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12793             Roo.log(e.body);
12794             //Roo.log(t.compiled);
12795             Roo.log(values);
12796             return '';
12797         }
12798     },
12799
12800    
12801
12802     applyTemplate : function(values){
12803         return this.master.compiled.call(this, values, {});
12804         //var s = this.subs;
12805     },
12806
12807     apply : function(){
12808         return this.applyTemplate.apply(this, arguments);
12809     }
12810
12811  });
12812
12813 Roo.DomTemplate.from = function(el){
12814     el = Roo.getDom(el);
12815     return new Roo.Domtemplate(el.value || el.innerHTML);
12816 };/*
12817  * Based on:
12818  * Ext JS Library 1.1.1
12819  * Copyright(c) 2006-2007, Ext JS, LLC.
12820  *
12821  * Originally Released Under LGPL - original licence link has changed is not relivant.
12822  *
12823  * Fork - LGPL
12824  * <script type="text/javascript">
12825  */
12826
12827 /**
12828  * @class Roo.util.DelayedTask
12829  * Provides a convenient method of performing setTimeout where a new
12830  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12831  * You can use this class to buffer
12832  * the keypress events for a certain number of milliseconds, and perform only if they stop
12833  * for that amount of time.
12834  * @constructor The parameters to this constructor serve as defaults and are not required.
12835  * @param {Function} fn (optional) The default function to timeout
12836  * @param {Object} scope (optional) The default scope of that timeout
12837  * @param {Array} args (optional) The default Array of arguments
12838  */
12839 Roo.util.DelayedTask = function(fn, scope, args){
12840     var id = null, d, t;
12841
12842     var call = function(){
12843         var now = new Date().getTime();
12844         if(now - t >= d){
12845             clearInterval(id);
12846             id = null;
12847             fn.apply(scope, args || []);
12848         }
12849     };
12850     /**
12851      * Cancels any pending timeout and queues a new one
12852      * @param {Number} delay The milliseconds to delay
12853      * @param {Function} newFn (optional) Overrides function passed to constructor
12854      * @param {Object} newScope (optional) Overrides scope passed to constructor
12855      * @param {Array} newArgs (optional) Overrides args passed to constructor
12856      */
12857     this.delay = function(delay, newFn, newScope, newArgs){
12858         if(id && delay != d){
12859             this.cancel();
12860         }
12861         d = delay;
12862         t = new Date().getTime();
12863         fn = newFn || fn;
12864         scope = newScope || scope;
12865         args = newArgs || args;
12866         if(!id){
12867             id = setInterval(call, d);
12868         }
12869     };
12870
12871     /**
12872      * Cancel the last queued timeout
12873      */
12874     this.cancel = function(){
12875         if(id){
12876             clearInterval(id);
12877             id = null;
12878         }
12879     };
12880 };/*
12881  * Based on:
12882  * Ext JS Library 1.1.1
12883  * Copyright(c) 2006-2007, Ext JS, LLC.
12884  *
12885  * Originally Released Under LGPL - original licence link has changed is not relivant.
12886  *
12887  * Fork - LGPL
12888  * <script type="text/javascript">
12889  */
12890  
12891  
12892 Roo.util.TaskRunner = function(interval){
12893     interval = interval || 10;
12894     var tasks = [], removeQueue = [];
12895     var id = 0;
12896     var running = false;
12897
12898     var stopThread = function(){
12899         running = false;
12900         clearInterval(id);
12901         id = 0;
12902     };
12903
12904     var startThread = function(){
12905         if(!running){
12906             running = true;
12907             id = setInterval(runTasks, interval);
12908         }
12909     };
12910
12911     var removeTask = function(task){
12912         removeQueue.push(task);
12913         if(task.onStop){
12914             task.onStop();
12915         }
12916     };
12917
12918     var runTasks = function(){
12919         if(removeQueue.length > 0){
12920             for(var i = 0, len = removeQueue.length; i < len; i++){
12921                 tasks.remove(removeQueue[i]);
12922             }
12923             removeQueue = [];
12924             if(tasks.length < 1){
12925                 stopThread();
12926                 return;
12927             }
12928         }
12929         var now = new Date().getTime();
12930         for(var i = 0, len = tasks.length; i < len; ++i){
12931             var t = tasks[i];
12932             var itime = now - t.taskRunTime;
12933             if(t.interval <= itime){
12934                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12935                 t.taskRunTime = now;
12936                 if(rt === false || t.taskRunCount === t.repeat){
12937                     removeTask(t);
12938                     return;
12939                 }
12940             }
12941             if(t.duration && t.duration <= (now - t.taskStartTime)){
12942                 removeTask(t);
12943             }
12944         }
12945     };
12946
12947     /**
12948      * Queues a new task.
12949      * @param {Object} task
12950      */
12951     this.start = function(task){
12952         tasks.push(task);
12953         task.taskStartTime = new Date().getTime();
12954         task.taskRunTime = 0;
12955         task.taskRunCount = 0;
12956         startThread();
12957         return task;
12958     };
12959
12960     this.stop = function(task){
12961         removeTask(task);
12962         return task;
12963     };
12964
12965     this.stopAll = function(){
12966         stopThread();
12967         for(var i = 0, len = tasks.length; i < len; i++){
12968             if(tasks[i].onStop){
12969                 tasks[i].onStop();
12970             }
12971         }
12972         tasks = [];
12973         removeQueue = [];
12974     };
12975 };
12976
12977 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12978  * Based on:
12979  * Ext JS Library 1.1.1
12980  * Copyright(c) 2006-2007, Ext JS, LLC.
12981  *
12982  * Originally Released Under LGPL - original licence link has changed is not relivant.
12983  *
12984  * Fork - LGPL
12985  * <script type="text/javascript">
12986  */
12987
12988  
12989 /**
12990  * @class Roo.util.MixedCollection
12991  * @extends Roo.util.Observable
12992  * A Collection class that maintains both numeric indexes and keys and exposes events.
12993  * @constructor
12994  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
12995  * collection (defaults to false)
12996  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
12997  * and return the key value for that item.  This is used when available to look up the key on items that
12998  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
12999  * equivalent to providing an implementation for the {@link #getKey} method.
13000  */
13001 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13002     this.items = [];
13003     this.map = {};
13004     this.keys = [];
13005     this.length = 0;
13006     this.addEvents({
13007         /**
13008          * @event clear
13009          * Fires when the collection is cleared.
13010          */
13011         "clear" : true,
13012         /**
13013          * @event add
13014          * Fires when an item is added to the collection.
13015          * @param {Number} index The index at which the item was added.
13016          * @param {Object} o The item added.
13017          * @param {String} key The key associated with the added item.
13018          */
13019         "add" : true,
13020         /**
13021          * @event replace
13022          * Fires when an item is replaced in the collection.
13023          * @param {String} key he key associated with the new added.
13024          * @param {Object} old The item being replaced.
13025          * @param {Object} new The new item.
13026          */
13027         "replace" : true,
13028         /**
13029          * @event remove
13030          * Fires when an item is removed from the collection.
13031          * @param {Object} o The item being removed.
13032          * @param {String} key (optional) The key associated with the removed item.
13033          */
13034         "remove" : true,
13035         "sort" : true
13036     });
13037     this.allowFunctions = allowFunctions === true;
13038     if(keyFn){
13039         this.getKey = keyFn;
13040     }
13041     Roo.util.MixedCollection.superclass.constructor.call(this);
13042 };
13043
13044 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13045     allowFunctions : false,
13046     
13047 /**
13048  * Adds an item to the collection.
13049  * @param {String} key The key to associate with the item
13050  * @param {Object} o The item to add.
13051  * @return {Object} The item added.
13052  */
13053     add : function(key, o){
13054         if(arguments.length == 1){
13055             o = arguments[0];
13056             key = this.getKey(o);
13057         }
13058         if(typeof key == "undefined" || key === null){
13059             this.length++;
13060             this.items.push(o);
13061             this.keys.push(null);
13062         }else{
13063             var old = this.map[key];
13064             if(old){
13065                 return this.replace(key, o);
13066             }
13067             this.length++;
13068             this.items.push(o);
13069             this.map[key] = o;
13070             this.keys.push(key);
13071         }
13072         this.fireEvent("add", this.length-1, o, key);
13073         return o;
13074     },
13075        
13076 /**
13077   * MixedCollection has a generic way to fetch keys if you implement getKey.
13078 <pre><code>
13079 // normal way
13080 var mc = new Roo.util.MixedCollection();
13081 mc.add(someEl.dom.id, someEl);
13082 mc.add(otherEl.dom.id, otherEl);
13083 //and so on
13084
13085 // using getKey
13086 var mc = new Roo.util.MixedCollection();
13087 mc.getKey = function(el){
13088    return el.dom.id;
13089 };
13090 mc.add(someEl);
13091 mc.add(otherEl);
13092
13093 // or via the constructor
13094 var mc = new Roo.util.MixedCollection(false, function(el){
13095    return el.dom.id;
13096 });
13097 mc.add(someEl);
13098 mc.add(otherEl);
13099 </code></pre>
13100  * @param o {Object} The item for which to find the key.
13101  * @return {Object} The key for the passed item.
13102  */
13103     getKey : function(o){
13104          return o.id; 
13105     },
13106    
13107 /**
13108  * Replaces an item in the collection.
13109  * @param {String} key The key associated with the item to replace, or the item to replace.
13110  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13111  * @return {Object}  The new item.
13112  */
13113     replace : function(key, o){
13114         if(arguments.length == 1){
13115             o = arguments[0];
13116             key = this.getKey(o);
13117         }
13118         var old = this.item(key);
13119         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13120              return this.add(key, o);
13121         }
13122         var index = this.indexOfKey(key);
13123         this.items[index] = o;
13124         this.map[key] = o;
13125         this.fireEvent("replace", key, old, o);
13126         return o;
13127     },
13128    
13129 /**
13130  * Adds all elements of an Array or an Object to the collection.
13131  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13132  * an Array of values, each of which are added to the collection.
13133  */
13134     addAll : function(objs){
13135         if(arguments.length > 1 || objs instanceof Array){
13136             var args = arguments.length > 1 ? arguments : objs;
13137             for(var i = 0, len = args.length; i < len; i++){
13138                 this.add(args[i]);
13139             }
13140         }else{
13141             for(var key in objs){
13142                 if(this.allowFunctions || typeof objs[key] != "function"){
13143                     this.add(key, objs[key]);
13144                 }
13145             }
13146         }
13147     },
13148    
13149 /**
13150  * Executes the specified function once for every item in the collection, passing each
13151  * item as the first and only parameter. returning false from the function will stop the iteration.
13152  * @param {Function} fn The function to execute for each item.
13153  * @param {Object} scope (optional) The scope in which to execute the function.
13154  */
13155     each : function(fn, scope){
13156         var items = [].concat(this.items); // each safe for removal
13157         for(var i = 0, len = items.length; i < len; i++){
13158             if(fn.call(scope || items[i], items[i], i, len) === false){
13159                 break;
13160             }
13161         }
13162     },
13163    
13164 /**
13165  * Executes the specified function once for every key in the collection, passing each
13166  * key, and its associated item as the first two parameters.
13167  * @param {Function} fn The function to execute for each item.
13168  * @param {Object} scope (optional) The scope in which to execute the function.
13169  */
13170     eachKey : function(fn, scope){
13171         for(var i = 0, len = this.keys.length; i < len; i++){
13172             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13173         }
13174     },
13175    
13176 /**
13177  * Returns the first item in the collection which elicits a true return value from the
13178  * passed selection function.
13179  * @param {Function} fn The selection function to execute for each item.
13180  * @param {Object} scope (optional) The scope in which to execute the function.
13181  * @return {Object} The first item in the collection which returned true from the selection function.
13182  */
13183     find : function(fn, scope){
13184         for(var i = 0, len = this.items.length; i < len; i++){
13185             if(fn.call(scope || window, this.items[i], this.keys[i])){
13186                 return this.items[i];
13187             }
13188         }
13189         return null;
13190     },
13191    
13192 /**
13193  * Inserts an item at the specified index in the collection.
13194  * @param {Number} index The index to insert the item at.
13195  * @param {String} key The key to associate with the new item, or the item itself.
13196  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13197  * @return {Object} The item inserted.
13198  */
13199     insert : function(index, key, o){
13200         if(arguments.length == 2){
13201             o = arguments[1];
13202             key = this.getKey(o);
13203         }
13204         if(index >= this.length){
13205             return this.add(key, o);
13206         }
13207         this.length++;
13208         this.items.splice(index, 0, o);
13209         if(typeof key != "undefined" && key != null){
13210             this.map[key] = o;
13211         }
13212         this.keys.splice(index, 0, key);
13213         this.fireEvent("add", index, o, key);
13214         return o;
13215     },
13216    
13217 /**
13218  * Removed an item from the collection.
13219  * @param {Object} o The item to remove.
13220  * @return {Object} The item removed.
13221  */
13222     remove : function(o){
13223         return this.removeAt(this.indexOf(o));
13224     },
13225    
13226 /**
13227  * Remove an item from a specified index in the collection.
13228  * @param {Number} index The index within the collection of the item to remove.
13229  */
13230     removeAt : function(index){
13231         if(index < this.length && index >= 0){
13232             this.length--;
13233             var o = this.items[index];
13234             this.items.splice(index, 1);
13235             var key = this.keys[index];
13236             if(typeof key != "undefined"){
13237                 delete this.map[key];
13238             }
13239             this.keys.splice(index, 1);
13240             this.fireEvent("remove", o, key);
13241         }
13242     },
13243    
13244 /**
13245  * Removed an item associated with the passed key fom the collection.
13246  * @param {String} key The key of the item to remove.
13247  */
13248     removeKey : function(key){
13249         return this.removeAt(this.indexOfKey(key));
13250     },
13251    
13252 /**
13253  * Returns the number of items in the collection.
13254  * @return {Number} the number of items in the collection.
13255  */
13256     getCount : function(){
13257         return this.length; 
13258     },
13259    
13260 /**
13261  * Returns index within the collection of the passed Object.
13262  * @param {Object} o The item to find the index of.
13263  * @return {Number} index of the item.
13264  */
13265     indexOf : function(o){
13266         if(!this.items.indexOf){
13267             for(var i = 0, len = this.items.length; i < len; i++){
13268                 if(this.items[i] == o) {
13269                     return i;
13270                 }
13271             }
13272             return -1;
13273         }else{
13274             return this.items.indexOf(o);
13275         }
13276     },
13277    
13278 /**
13279  * Returns index within the collection of the passed key.
13280  * @param {String} key The key to find the index of.
13281  * @return {Number} index of the key.
13282  */
13283     indexOfKey : function(key){
13284         if(!this.keys.indexOf){
13285             for(var i = 0, len = this.keys.length; i < len; i++){
13286                 if(this.keys[i] == key) {
13287                     return i;
13288                 }
13289             }
13290             return -1;
13291         }else{
13292             return this.keys.indexOf(key);
13293         }
13294     },
13295    
13296 /**
13297  * Returns the item associated with the passed key OR index. Key has priority over index.
13298  * @param {String/Number} key The key or index of the item.
13299  * @return {Object} The item associated with the passed key.
13300  */
13301     item : function(key){
13302         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13303         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13304     },
13305     
13306 /**
13307  * Returns the item at the specified index.
13308  * @param {Number} index The index of the item.
13309  * @return {Object}
13310  */
13311     itemAt : function(index){
13312         return this.items[index];
13313     },
13314     
13315 /**
13316  * Returns the item associated with the passed key.
13317  * @param {String/Number} key The key of the item.
13318  * @return {Object} The item associated with the passed key.
13319  */
13320     key : function(key){
13321         return this.map[key];
13322     },
13323    
13324 /**
13325  * Returns true if the collection contains the passed Object as an item.
13326  * @param {Object} o  The Object to look for in the collection.
13327  * @return {Boolean} True if the collection contains the Object as an item.
13328  */
13329     contains : function(o){
13330         return this.indexOf(o) != -1;
13331     },
13332    
13333 /**
13334  * Returns true if the collection contains the passed Object as a key.
13335  * @param {String} key The key to look for in the collection.
13336  * @return {Boolean} True if the collection contains the Object as a key.
13337  */
13338     containsKey : function(key){
13339         return typeof this.map[key] != "undefined";
13340     },
13341    
13342 /**
13343  * Removes all items from the collection.
13344  */
13345     clear : function(){
13346         this.length = 0;
13347         this.items = [];
13348         this.keys = [];
13349         this.map = {};
13350         this.fireEvent("clear");
13351     },
13352    
13353 /**
13354  * Returns the first item in the collection.
13355  * @return {Object} the first item in the collection..
13356  */
13357     first : function(){
13358         return this.items[0]; 
13359     },
13360    
13361 /**
13362  * Returns the last item in the collection.
13363  * @return {Object} the last item in the collection..
13364  */
13365     last : function(){
13366         return this.items[this.length-1];   
13367     },
13368     
13369     _sort : function(property, dir, fn){
13370         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13371         fn = fn || function(a, b){
13372             return a-b;
13373         };
13374         var c = [], k = this.keys, items = this.items;
13375         for(var i = 0, len = items.length; i < len; i++){
13376             c[c.length] = {key: k[i], value: items[i], index: i};
13377         }
13378         c.sort(function(a, b){
13379             var v = fn(a[property], b[property]) * dsc;
13380             if(v == 0){
13381                 v = (a.index < b.index ? -1 : 1);
13382             }
13383             return v;
13384         });
13385         for(var i = 0, len = c.length; i < len; i++){
13386             items[i] = c[i].value;
13387             k[i] = c[i].key;
13388         }
13389         this.fireEvent("sort", this);
13390     },
13391     
13392     /**
13393      * Sorts this collection with the passed comparison function
13394      * @param {String} direction (optional) "ASC" or "DESC"
13395      * @param {Function} fn (optional) comparison function
13396      */
13397     sort : function(dir, fn){
13398         this._sort("value", dir, fn);
13399     },
13400     
13401     /**
13402      * Sorts this collection by keys
13403      * @param {String} direction (optional) "ASC" or "DESC"
13404      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13405      */
13406     keySort : function(dir, fn){
13407         this._sort("key", dir, fn || function(a, b){
13408             return String(a).toUpperCase()-String(b).toUpperCase();
13409         });
13410     },
13411     
13412     /**
13413      * Returns a range of items in this collection
13414      * @param {Number} startIndex (optional) defaults to 0
13415      * @param {Number} endIndex (optional) default to the last item
13416      * @return {Array} An array of items
13417      */
13418     getRange : function(start, end){
13419         var items = this.items;
13420         if(items.length < 1){
13421             return [];
13422         }
13423         start = start || 0;
13424         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13425         var r = [];
13426         if(start <= end){
13427             for(var i = start; i <= end; i++) {
13428                     r[r.length] = items[i];
13429             }
13430         }else{
13431             for(var i = start; i >= end; i--) {
13432                     r[r.length] = items[i];
13433             }
13434         }
13435         return r;
13436     },
13437         
13438     /**
13439      * Filter the <i>objects</i> in this collection by a specific property. 
13440      * Returns a new collection that has been filtered.
13441      * @param {String} property A property on your objects
13442      * @param {String/RegExp} value Either string that the property values 
13443      * should start with or a RegExp to test against the property
13444      * @return {MixedCollection} The new filtered collection
13445      */
13446     filter : function(property, value){
13447         if(!value.exec){ // not a regex
13448             value = String(value);
13449             if(value.length == 0){
13450                 return this.clone();
13451             }
13452             value = new RegExp("^" + Roo.escapeRe(value), "i");
13453         }
13454         return this.filterBy(function(o){
13455             return o && value.test(o[property]);
13456         });
13457         },
13458     
13459     /**
13460      * Filter by a function. * Returns a new collection that has been filtered.
13461      * The passed function will be called with each 
13462      * object in the collection. If the function returns true, the value is included 
13463      * otherwise it is filtered.
13464      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13465      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13466      * @return {MixedCollection} The new filtered collection
13467      */
13468     filterBy : function(fn, scope){
13469         var r = new Roo.util.MixedCollection();
13470         r.getKey = this.getKey;
13471         var k = this.keys, it = this.items;
13472         for(var i = 0, len = it.length; i < len; i++){
13473             if(fn.call(scope||this, it[i], k[i])){
13474                                 r.add(k[i], it[i]);
13475                         }
13476         }
13477         return r;
13478     },
13479     
13480     /**
13481      * Creates a duplicate of this collection
13482      * @return {MixedCollection}
13483      */
13484     clone : function(){
13485         var r = new Roo.util.MixedCollection();
13486         var k = this.keys, it = this.items;
13487         for(var i = 0, len = it.length; i < len; i++){
13488             r.add(k[i], it[i]);
13489         }
13490         r.getKey = this.getKey;
13491         return r;
13492     }
13493 });
13494 /**
13495  * Returns the item associated with the passed key or index.
13496  * @method
13497  * @param {String/Number} key The key or index of the item.
13498  * @return {Object} The item associated with the passed key.
13499  */
13500 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13501  * Based on:
13502  * Ext JS Library 1.1.1
13503  * Copyright(c) 2006-2007, Ext JS, LLC.
13504  *
13505  * Originally Released Under LGPL - original licence link has changed is not relivant.
13506  *
13507  * Fork - LGPL
13508  * <script type="text/javascript">
13509  */
13510 /**
13511  * @class Roo.util.JSON
13512  * Modified version of Douglas Crockford"s json.js that doesn"t
13513  * mess with the Object prototype 
13514  * http://www.json.org/js.html
13515  * @singleton
13516  */
13517 Roo.util.JSON = new (function(){
13518     var useHasOwn = {}.hasOwnProperty ? true : false;
13519     
13520     // crashes Safari in some instances
13521     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13522     
13523     var pad = function(n) {
13524         return n < 10 ? "0" + n : n;
13525     };
13526     
13527     var m = {
13528         "\b": '\\b',
13529         "\t": '\\t',
13530         "\n": '\\n',
13531         "\f": '\\f',
13532         "\r": '\\r',
13533         '"' : '\\"',
13534         "\\": '\\\\'
13535     };
13536
13537     var encodeString = function(s){
13538         if (/["\\\x00-\x1f]/.test(s)) {
13539             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13540                 var c = m[b];
13541                 if(c){
13542                     return c;
13543                 }
13544                 c = b.charCodeAt();
13545                 return "\\u00" +
13546                     Math.floor(c / 16).toString(16) +
13547                     (c % 16).toString(16);
13548             }) + '"';
13549         }
13550         return '"' + s + '"';
13551     };
13552     
13553     var encodeArray = function(o){
13554         var a = ["["], b, i, l = o.length, v;
13555             for (i = 0; i < l; i += 1) {
13556                 v = o[i];
13557                 switch (typeof v) {
13558                     case "undefined":
13559                     case "function":
13560                     case "unknown":
13561                         break;
13562                     default:
13563                         if (b) {
13564                             a.push(',');
13565                         }
13566                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13567                         b = true;
13568                 }
13569             }
13570             a.push("]");
13571             return a.join("");
13572     };
13573     
13574     var encodeDate = function(o){
13575         return '"' + o.getFullYear() + "-" +
13576                 pad(o.getMonth() + 1) + "-" +
13577                 pad(o.getDate()) + "T" +
13578                 pad(o.getHours()) + ":" +
13579                 pad(o.getMinutes()) + ":" +
13580                 pad(o.getSeconds()) + '"';
13581     };
13582     
13583     /**
13584      * Encodes an Object, Array or other value
13585      * @param {Mixed} o The variable to encode
13586      * @return {String} The JSON string
13587      */
13588     this.encode = function(o)
13589     {
13590         // should this be extended to fully wrap stringify..
13591         
13592         if(typeof o == "undefined" || o === null){
13593             return "null";
13594         }else if(o instanceof Array){
13595             return encodeArray(o);
13596         }else if(o instanceof Date){
13597             return encodeDate(o);
13598         }else if(typeof o == "string"){
13599             return encodeString(o);
13600         }else if(typeof o == "number"){
13601             return isFinite(o) ? String(o) : "null";
13602         }else if(typeof o == "boolean"){
13603             return String(o);
13604         }else {
13605             var a = ["{"], b, i, v;
13606             for (i in o) {
13607                 if(!useHasOwn || o.hasOwnProperty(i)) {
13608                     v = o[i];
13609                     switch (typeof v) {
13610                     case "undefined":
13611                     case "function":
13612                     case "unknown":
13613                         break;
13614                     default:
13615                         if(b){
13616                             a.push(',');
13617                         }
13618                         a.push(this.encode(i), ":",
13619                                 v === null ? "null" : this.encode(v));
13620                         b = true;
13621                     }
13622                 }
13623             }
13624             a.push("}");
13625             return a.join("");
13626         }
13627     };
13628     
13629     /**
13630      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13631      * @param {String} json The JSON string
13632      * @return {Object} The resulting object
13633      */
13634     this.decode = function(json){
13635         
13636         return  /** eval:var:json */ eval("(" + json + ')');
13637     };
13638 })();
13639 /** 
13640  * Shorthand for {@link Roo.util.JSON#encode}
13641  * @member Roo encode 
13642  * @method */
13643 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#decode}
13646  * @member Roo decode 
13647  * @method */
13648 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13649 /*
13650  * Based on:
13651  * Ext JS Library 1.1.1
13652  * Copyright(c) 2006-2007, Ext JS, LLC.
13653  *
13654  * Originally Released Under LGPL - original licence link has changed is not relivant.
13655  *
13656  * Fork - LGPL
13657  * <script type="text/javascript">
13658  */
13659  
13660 /**
13661  * @class Roo.util.Format
13662  * Reusable data formatting functions
13663  * @singleton
13664  */
13665 Roo.util.Format = function(){
13666     var trimRe = /^\s+|\s+$/g;
13667     return {
13668         /**
13669          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13670          * @param {String} value The string to truncate
13671          * @param {Number} length The maximum length to allow before truncating
13672          * @return {String} The converted text
13673          */
13674         ellipsis : function(value, len){
13675             if(value && value.length > len){
13676                 return value.substr(0, len-3)+"...";
13677             }
13678             return value;
13679         },
13680
13681         /**
13682          * Checks a reference and converts it to empty string if it is undefined
13683          * @param {Mixed} value Reference to check
13684          * @return {Mixed} Empty string if converted, otherwise the original value
13685          */
13686         undef : function(value){
13687             return typeof value != "undefined" ? value : "";
13688         },
13689
13690         /**
13691          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13692          * @param {String} value The string to encode
13693          * @return {String} The encoded text
13694          */
13695         htmlEncode : function(value){
13696             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13697         },
13698
13699         /**
13700          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13701          * @param {String} value The string to decode
13702          * @return {String} The decoded text
13703          */
13704         htmlDecode : function(value){
13705             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13706         },
13707
13708         /**
13709          * Trims any whitespace from either side of a string
13710          * @param {String} value The text to trim
13711          * @return {String} The trimmed text
13712          */
13713         trim : function(value){
13714             return String(value).replace(trimRe, "");
13715         },
13716
13717         /**
13718          * Returns a substring from within an original string
13719          * @param {String} value The original text
13720          * @param {Number} start The start index of the substring
13721          * @param {Number} length The length of the substring
13722          * @return {String} The substring
13723          */
13724         substr : function(value, start, length){
13725             return String(value).substr(start, length);
13726         },
13727
13728         /**
13729          * Converts a string to all lower case letters
13730          * @param {String} value The text to convert
13731          * @return {String} The converted text
13732          */
13733         lowercase : function(value){
13734             return String(value).toLowerCase();
13735         },
13736
13737         /**
13738          * Converts a string to all upper case letters
13739          * @param {String} value The text to convert
13740          * @return {String} The converted text
13741          */
13742         uppercase : function(value){
13743             return String(value).toUpperCase();
13744         },
13745
13746         /**
13747          * Converts the first character only of a string to upper case
13748          * @param {String} value The text to convert
13749          * @return {String} The converted text
13750          */
13751         capitalize : function(value){
13752             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13753         },
13754
13755         // private
13756         call : function(value, fn){
13757             if(arguments.length > 2){
13758                 var args = Array.prototype.slice.call(arguments, 2);
13759                 args.unshift(value);
13760                  
13761                 return /** eval:var:value */  eval(fn).apply(window, args);
13762             }else{
13763                 /** eval:var:value */
13764                 return /** eval:var:value */ eval(fn).call(window, value);
13765             }
13766         },
13767
13768        
13769         /**
13770          * safer version of Math.toFixed..??/
13771          * @param {Number/String} value The numeric value to format
13772          * @param {Number/String} value Decimal places 
13773          * @return {String} The formatted currency string
13774          */
13775         toFixed : function(v, n)
13776         {
13777             // why not use to fixed - precision is buggered???
13778             if (!n) {
13779                 return Math.round(v-0);
13780             }
13781             var fact = Math.pow(10,n+1);
13782             v = (Math.round((v-0)*fact))/fact;
13783             var z = (''+fact).substring(2);
13784             if (v == Math.floor(v)) {
13785                 return Math.floor(v) + '.' + z;
13786             }
13787             
13788             // now just padd decimals..
13789             var ps = String(v).split('.');
13790             var fd = (ps[1] + z);
13791             var r = fd.substring(0,n); 
13792             var rm = fd.substring(n); 
13793             if (rm < 5) {
13794                 return ps[0] + '.' + r;
13795             }
13796             r*=1; // turn it into a number;
13797             r++;
13798             if (String(r).length != n) {
13799                 ps[0]*=1;
13800                 ps[0]++;
13801                 r = String(r).substring(1); // chop the end off.
13802             }
13803             
13804             return ps[0] + '.' + r;
13805              
13806         },
13807         
13808         /**
13809          * Format a number as US currency
13810          * @param {Number/String} value The numeric value to format
13811          * @return {String} The formatted currency string
13812          */
13813         usMoney : function(v){
13814             return '$' + Roo.util.Format.number(v);
13815         },
13816         
13817         /**
13818          * Format a number
13819          * eventually this should probably emulate php's number_format
13820          * @param {Number/String} value The numeric value to format
13821          * @param {Number} decimals number of decimal places
13822          * @param {String} delimiter for thousands (default comma)
13823          * @return {String} The formatted currency string
13824          */
13825         number : function(v, decimals, thousandsDelimiter)
13826         {
13827             // multiply and round.
13828             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13829             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13830             
13831             var mul = Math.pow(10, decimals);
13832             var zero = String(mul).substring(1);
13833             v = (Math.round((v-0)*mul))/mul;
13834             
13835             // if it's '0' number.. then
13836             
13837             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13838             v = String(v);
13839             var ps = v.split('.');
13840             var whole = ps[0];
13841             
13842             var r = /(\d+)(\d{3})/;
13843             // add comma's
13844             
13845             if(thousandsDelimiter.length != 0) {
13846                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13847             } 
13848             
13849             var sub = ps[1] ?
13850                     // has decimals..
13851                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13852                     // does not have decimals
13853                     (decimals ? ('.' + zero) : '');
13854             
13855             
13856             return whole + sub ;
13857         },
13858         
13859         /**
13860          * Parse a value into a formatted date using the specified format pattern.
13861          * @param {Mixed} value The value to format
13862          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13863          * @return {String} The formatted date string
13864          */
13865         date : function(v, format){
13866             if(!v){
13867                 return "";
13868             }
13869             if(!(v instanceof Date)){
13870                 v = new Date(Date.parse(v));
13871             }
13872             return v.dateFormat(format || Roo.util.Format.defaults.date);
13873         },
13874
13875         /**
13876          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13877          * @param {String} format Any valid date format string
13878          * @return {Function} The date formatting function
13879          */
13880         dateRenderer : function(format){
13881             return function(v){
13882                 return Roo.util.Format.date(v, format);  
13883             };
13884         },
13885
13886         // private
13887         stripTagsRE : /<\/?[^>]+>/gi,
13888         
13889         /**
13890          * Strips all HTML tags
13891          * @param {Mixed} value The text from which to strip tags
13892          * @return {String} The stripped text
13893          */
13894         stripTags : function(v){
13895             return !v ? v : String(v).replace(this.stripTagsRE, "");
13896         }
13897     };
13898 }();
13899 Roo.util.Format.defaults = {
13900     date : 'd/M/Y'
13901 };/*
13902  * Based on:
13903  * Ext JS Library 1.1.1
13904  * Copyright(c) 2006-2007, Ext JS, LLC.
13905  *
13906  * Originally Released Under LGPL - original licence link has changed is not relivant.
13907  *
13908  * Fork - LGPL
13909  * <script type="text/javascript">
13910  */
13911
13912
13913  
13914
13915 /**
13916  * @class Roo.MasterTemplate
13917  * @extends Roo.Template
13918  * Provides a template that can have child templates. The syntax is:
13919 <pre><code>
13920 var t = new Roo.MasterTemplate(
13921         '&lt;select name="{name}"&gt;',
13922                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13923         '&lt;/select&gt;'
13924 );
13925 t.add('options', {value: 'foo', text: 'bar'});
13926 // or you can add multiple child elements in one shot
13927 t.addAll('options', [
13928     {value: 'foo', text: 'bar'},
13929     {value: 'foo2', text: 'bar2'},
13930     {value: 'foo3', text: 'bar3'}
13931 ]);
13932 // then append, applying the master template values
13933 t.append('my-form', {name: 'my-select'});
13934 </code></pre>
13935 * A name attribute for the child template is not required if you have only one child
13936 * template or you want to refer to them by index.
13937  */
13938 Roo.MasterTemplate = function(){
13939     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13940     this.originalHtml = this.html;
13941     var st = {};
13942     var m, re = this.subTemplateRe;
13943     re.lastIndex = 0;
13944     var subIndex = 0;
13945     while(m = re.exec(this.html)){
13946         var name = m[1], content = m[2];
13947         st[subIndex] = {
13948             name: name,
13949             index: subIndex,
13950             buffer: [],
13951             tpl : new Roo.Template(content)
13952         };
13953         if(name){
13954             st[name] = st[subIndex];
13955         }
13956         st[subIndex].tpl.compile();
13957         st[subIndex].tpl.call = this.call.createDelegate(this);
13958         subIndex++;
13959     }
13960     this.subCount = subIndex;
13961     this.subs = st;
13962 };
13963 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13964     /**
13965     * The regular expression used to match sub templates
13966     * @type RegExp
13967     * @property
13968     */
13969     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13970
13971     /**
13972      * Applies the passed values to a child template.
13973      * @param {String/Number} name (optional) The name or index of the child template
13974      * @param {Array/Object} values The values to be applied to the template
13975      * @return {MasterTemplate} this
13976      */
13977      add : function(name, values){
13978         if(arguments.length == 1){
13979             values = arguments[0];
13980             name = 0;
13981         }
13982         var s = this.subs[name];
13983         s.buffer[s.buffer.length] = s.tpl.apply(values);
13984         return this;
13985     },
13986
13987     /**
13988      * Applies all the passed values to a child template.
13989      * @param {String/Number} name (optional) The name or index of the child template
13990      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13991      * @param {Boolean} reset (optional) True to reset the template first
13992      * @return {MasterTemplate} this
13993      */
13994     fill : function(name, values, reset){
13995         var a = arguments;
13996         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
13997             values = a[0];
13998             name = 0;
13999             reset = a[1];
14000         }
14001         if(reset){
14002             this.reset();
14003         }
14004         for(var i = 0, len = values.length; i < len; i++){
14005             this.add(name, values[i]);
14006         }
14007         return this;
14008     },
14009
14010     /**
14011      * Resets the template for reuse
14012      * @return {MasterTemplate} this
14013      */
14014      reset : function(){
14015         var s = this.subs;
14016         for(var i = 0; i < this.subCount; i++){
14017             s[i].buffer = [];
14018         }
14019         return this;
14020     },
14021
14022     applyTemplate : function(values){
14023         var s = this.subs;
14024         var replaceIndex = -1;
14025         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14026             return s[++replaceIndex].buffer.join("");
14027         });
14028         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14029     },
14030
14031     apply : function(){
14032         return this.applyTemplate.apply(this, arguments);
14033     },
14034
14035     compile : function(){return this;}
14036 });
14037
14038 /**
14039  * Alias for fill().
14040  * @method
14041  */
14042 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14043  /**
14044  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14045  * var tpl = Roo.MasterTemplate.from('element-id');
14046  * @param {String/HTMLElement} el
14047  * @param {Object} config
14048  * @static
14049  */
14050 Roo.MasterTemplate.from = function(el, config){
14051     el = Roo.getDom(el);
14052     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14053 };/*
14054  * Based on:
14055  * Ext JS Library 1.1.1
14056  * Copyright(c) 2006-2007, Ext JS, LLC.
14057  *
14058  * Originally Released Under LGPL - original licence link has changed is not relivant.
14059  *
14060  * Fork - LGPL
14061  * <script type="text/javascript">
14062  */
14063
14064  
14065 /**
14066  * @class Roo.util.CSS
14067  * Utility class for manipulating CSS rules
14068  * @singleton
14069  */
14070 Roo.util.CSS = function(){
14071         var rules = null;
14072         var doc = document;
14073
14074     var camelRe = /(-[a-z])/gi;
14075     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14076
14077    return {
14078    /**
14079     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14080     * tag and appended to the HEAD of the document.
14081     * @param {String|Object} cssText The text containing the css rules
14082     * @param {String} id An id to add to the stylesheet for later removal
14083     * @return {StyleSheet}
14084     */
14085     createStyleSheet : function(cssText, id){
14086         var ss;
14087         var head = doc.getElementsByTagName("head")[0];
14088         var nrules = doc.createElement("style");
14089         nrules.setAttribute("type", "text/css");
14090         if(id){
14091             nrules.setAttribute("id", id);
14092         }
14093         if (typeof(cssText) != 'string') {
14094             // support object maps..
14095             // not sure if this a good idea.. 
14096             // perhaps it should be merged with the general css handling
14097             // and handle js style props.
14098             var cssTextNew = [];
14099             for(var n in cssText) {
14100                 var citems = [];
14101                 for(var k in cssText[n]) {
14102                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14103                 }
14104                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14105                 
14106             }
14107             cssText = cssTextNew.join("\n");
14108             
14109         }
14110        
14111        
14112        if(Roo.isIE){
14113            head.appendChild(nrules);
14114            ss = nrules.styleSheet;
14115            ss.cssText = cssText;
14116        }else{
14117            try{
14118                 nrules.appendChild(doc.createTextNode(cssText));
14119            }catch(e){
14120                nrules.cssText = cssText; 
14121            }
14122            head.appendChild(nrules);
14123            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14124        }
14125        this.cacheStyleSheet(ss);
14126        return ss;
14127    },
14128
14129    /**
14130     * Removes a style or link tag by id
14131     * @param {String} id The id of the tag
14132     */
14133    removeStyleSheet : function(id){
14134        var existing = doc.getElementById(id);
14135        if(existing){
14136            existing.parentNode.removeChild(existing);
14137        }
14138    },
14139
14140    /**
14141     * Dynamically swaps an existing stylesheet reference for a new one
14142     * @param {String} id The id of an existing link tag to remove
14143     * @param {String} url The href of the new stylesheet to include
14144     */
14145    swapStyleSheet : function(id, url){
14146        this.removeStyleSheet(id);
14147        var ss = doc.createElement("link");
14148        ss.setAttribute("rel", "stylesheet");
14149        ss.setAttribute("type", "text/css");
14150        ss.setAttribute("id", id);
14151        ss.setAttribute("href", url);
14152        doc.getElementsByTagName("head")[0].appendChild(ss);
14153    },
14154    
14155    /**
14156     * Refresh the rule cache if you have dynamically added stylesheets
14157     * @return {Object} An object (hash) of rules indexed by selector
14158     */
14159    refreshCache : function(){
14160        return this.getRules(true);
14161    },
14162
14163    // private
14164    cacheStyleSheet : function(stylesheet){
14165        if(!rules){
14166            rules = {};
14167        }
14168        try{// try catch for cross domain access issue
14169            var ssRules = stylesheet.cssRules || stylesheet.rules;
14170            for(var j = ssRules.length-1; j >= 0; --j){
14171                rules[ssRules[j].selectorText] = ssRules[j];
14172            }
14173        }catch(e){}
14174    },
14175    
14176    /**
14177     * Gets all css rules for the document
14178     * @param {Boolean} refreshCache true to refresh the internal cache
14179     * @return {Object} An object (hash) of rules indexed by selector
14180     */
14181    getRules : function(refreshCache){
14182                 if(rules == null || refreshCache){
14183                         rules = {};
14184                         var ds = doc.styleSheets;
14185                         for(var i =0, len = ds.length; i < len; i++){
14186                             try{
14187                         this.cacheStyleSheet(ds[i]);
14188                     }catch(e){} 
14189                 }
14190                 }
14191                 return rules;
14192         },
14193         
14194         /**
14195     * Gets an an individual CSS rule by selector(s)
14196     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14197     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14198     * @return {CSSRule} The CSS rule or null if one is not found
14199     */
14200    getRule : function(selector, refreshCache){
14201                 var rs = this.getRules(refreshCache);
14202                 if(!(selector instanceof Array)){
14203                     return rs[selector];
14204                 }
14205                 for(var i = 0; i < selector.length; i++){
14206                         if(rs[selector[i]]){
14207                                 return rs[selector[i]];
14208                         }
14209                 }
14210                 return null;
14211         },
14212         
14213         
14214         /**
14215     * Updates a rule property
14216     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14217     * @param {String} property The css property
14218     * @param {String} value The new value for the property
14219     * @return {Boolean} true If a rule was found and updated
14220     */
14221    updateRule : function(selector, property, value){
14222                 if(!(selector instanceof Array)){
14223                         var rule = this.getRule(selector);
14224                         if(rule){
14225                                 rule.style[property.replace(camelRe, camelFn)] = value;
14226                                 return true;
14227                         }
14228                 }else{
14229                         for(var i = 0; i < selector.length; i++){
14230                                 if(this.updateRule(selector[i], property, value)){
14231                                         return true;
14232                                 }
14233                         }
14234                 }
14235                 return false;
14236         }
14237    };   
14238 }();/*
14239  * Based on:
14240  * Ext JS Library 1.1.1
14241  * Copyright(c) 2006-2007, Ext JS, LLC.
14242  *
14243  * Originally Released Under LGPL - original licence link has changed is not relivant.
14244  *
14245  * Fork - LGPL
14246  * <script type="text/javascript">
14247  */
14248
14249  
14250
14251 /**
14252  * @class Roo.util.ClickRepeater
14253  * @extends Roo.util.Observable
14254  * 
14255  * A wrapper class which can be applied to any element. Fires a "click" event while the
14256  * mouse is pressed. The interval between firings may be specified in the config but
14257  * defaults to 10 milliseconds.
14258  * 
14259  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14260  * 
14261  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14262  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14263  * Similar to an autorepeat key delay.
14264  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14265  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14266  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14267  *           "interval" and "delay" are ignored. "immediate" is honored.
14268  * @cfg {Boolean} preventDefault True to prevent the default click event
14269  * @cfg {Boolean} stopDefault True to stop the default click event
14270  * 
14271  * @history
14272  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14273  *     2007-02-02 jvs Renamed to ClickRepeater
14274  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14275  *
14276  *  @constructor
14277  * @param {String/HTMLElement/Element} el The element to listen on
14278  * @param {Object} config
14279  **/
14280 Roo.util.ClickRepeater = function(el, config)
14281 {
14282     this.el = Roo.get(el);
14283     this.el.unselectable();
14284
14285     Roo.apply(this, config);
14286
14287     this.addEvents({
14288     /**
14289      * @event mousedown
14290      * Fires when the mouse button is depressed.
14291      * @param {Roo.util.ClickRepeater} this
14292      */
14293         "mousedown" : true,
14294     /**
14295      * @event click
14296      * Fires on a specified interval during the time the element is pressed.
14297      * @param {Roo.util.ClickRepeater} this
14298      */
14299         "click" : true,
14300     /**
14301      * @event mouseup
14302      * Fires when the mouse key is released.
14303      * @param {Roo.util.ClickRepeater} this
14304      */
14305         "mouseup" : true
14306     });
14307
14308     this.el.on("mousedown", this.handleMouseDown, this);
14309     if(this.preventDefault || this.stopDefault){
14310         this.el.on("click", function(e){
14311             if(this.preventDefault){
14312                 e.preventDefault();
14313             }
14314             if(this.stopDefault){
14315                 e.stopEvent();
14316             }
14317         }, this);
14318     }
14319
14320     // allow inline handler
14321     if(this.handler){
14322         this.on("click", this.handler,  this.scope || this);
14323     }
14324
14325     Roo.util.ClickRepeater.superclass.constructor.call(this);
14326 };
14327
14328 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14329     interval : 20,
14330     delay: 250,
14331     preventDefault : true,
14332     stopDefault : false,
14333     timer : 0,
14334
14335     // private
14336     handleMouseDown : function(){
14337         clearTimeout(this.timer);
14338         this.el.blur();
14339         if(this.pressClass){
14340             this.el.addClass(this.pressClass);
14341         }
14342         this.mousedownTime = new Date();
14343
14344         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14345         this.el.on("mouseout", this.handleMouseOut, this);
14346
14347         this.fireEvent("mousedown", this);
14348         this.fireEvent("click", this);
14349         
14350         this.timer = this.click.defer(this.delay || this.interval, this);
14351     },
14352
14353     // private
14354     click : function(){
14355         this.fireEvent("click", this);
14356         this.timer = this.click.defer(this.getInterval(), this);
14357     },
14358
14359     // private
14360     getInterval: function(){
14361         if(!this.accelerate){
14362             return this.interval;
14363         }
14364         var pressTime = this.mousedownTime.getElapsed();
14365         if(pressTime < 500){
14366             return 400;
14367         }else if(pressTime < 1700){
14368             return 320;
14369         }else if(pressTime < 2600){
14370             return 250;
14371         }else if(pressTime < 3500){
14372             return 180;
14373         }else if(pressTime < 4400){
14374             return 140;
14375         }else if(pressTime < 5300){
14376             return 80;
14377         }else if(pressTime < 6200){
14378             return 50;
14379         }else{
14380             return 10;
14381         }
14382     },
14383
14384     // private
14385     handleMouseOut : function(){
14386         clearTimeout(this.timer);
14387         if(this.pressClass){
14388             this.el.removeClass(this.pressClass);
14389         }
14390         this.el.on("mouseover", this.handleMouseReturn, this);
14391     },
14392
14393     // private
14394     handleMouseReturn : function(){
14395         this.el.un("mouseover", this.handleMouseReturn);
14396         if(this.pressClass){
14397             this.el.addClass(this.pressClass);
14398         }
14399         this.click();
14400     },
14401
14402     // private
14403     handleMouseUp : function(){
14404         clearTimeout(this.timer);
14405         this.el.un("mouseover", this.handleMouseReturn);
14406         this.el.un("mouseout", this.handleMouseOut);
14407         Roo.get(document).un("mouseup", this.handleMouseUp);
14408         this.el.removeClass(this.pressClass);
14409         this.fireEvent("mouseup", this);
14410     }
14411 });/*
14412  * Based on:
14413  * Ext JS Library 1.1.1
14414  * Copyright(c) 2006-2007, Ext JS, LLC.
14415  *
14416  * Originally Released Under LGPL - original licence link has changed is not relivant.
14417  *
14418  * Fork - LGPL
14419  * <script type="text/javascript">
14420  */
14421
14422  
14423 /**
14424  * @class Roo.KeyNav
14425  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14426  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14427  * way to implement custom navigation schemes for any UI component.</p>
14428  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14429  * pageUp, pageDown, del, home, end.  Usage:</p>
14430  <pre><code>
14431 var nav = new Roo.KeyNav("my-element", {
14432     "left" : function(e){
14433         this.moveLeft(e.ctrlKey);
14434     },
14435     "right" : function(e){
14436         this.moveRight(e.ctrlKey);
14437     },
14438     "enter" : function(e){
14439         this.save();
14440     },
14441     scope : this
14442 });
14443 </code></pre>
14444  * @constructor
14445  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14446  * @param {Object} config The config
14447  */
14448 Roo.KeyNav = function(el, config){
14449     this.el = Roo.get(el);
14450     Roo.apply(this, config);
14451     if(!this.disabled){
14452         this.disabled = true;
14453         this.enable();
14454     }
14455 };
14456
14457 Roo.KeyNav.prototype = {
14458     /**
14459      * @cfg {Boolean} disabled
14460      * True to disable this KeyNav instance (defaults to false)
14461      */
14462     disabled : false,
14463     /**
14464      * @cfg {String} defaultEventAction
14465      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14466      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14467      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14468      */
14469     defaultEventAction: "stopEvent",
14470     /**
14471      * @cfg {Boolean} forceKeyDown
14472      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14473      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14474      * handle keydown instead of keypress.
14475      */
14476     forceKeyDown : false,
14477
14478     // private
14479     prepareEvent : function(e){
14480         var k = e.getKey();
14481         var h = this.keyToHandler[k];
14482         //if(h && this[h]){
14483         //    e.stopPropagation();
14484         //}
14485         if(Roo.isSafari && h && k >= 37 && k <= 40){
14486             e.stopEvent();
14487         }
14488     },
14489
14490     // private
14491     relay : function(e){
14492         var k = e.getKey();
14493         var h = this.keyToHandler[k];
14494         if(h && this[h]){
14495             if(this.doRelay(e, this[h], h) !== true){
14496                 e[this.defaultEventAction]();
14497             }
14498         }
14499     },
14500
14501     // private
14502     doRelay : function(e, h, hname){
14503         return h.call(this.scope || this, e);
14504     },
14505
14506     // possible handlers
14507     enter : false,
14508     left : false,
14509     right : false,
14510     up : false,
14511     down : false,
14512     tab : false,
14513     esc : false,
14514     pageUp : false,
14515     pageDown : false,
14516     del : false,
14517     home : false,
14518     end : false,
14519
14520     // quick lookup hash
14521     keyToHandler : {
14522         37 : "left",
14523         39 : "right",
14524         38 : "up",
14525         40 : "down",
14526         33 : "pageUp",
14527         34 : "pageDown",
14528         46 : "del",
14529         36 : "home",
14530         35 : "end",
14531         13 : "enter",
14532         27 : "esc",
14533         9  : "tab"
14534     },
14535
14536         /**
14537          * Enable this KeyNav
14538          */
14539         enable: function(){
14540                 if(this.disabled){
14541             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14542             // the EventObject will normalize Safari automatically
14543             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14544                 this.el.on("keydown", this.relay,  this);
14545             }else{
14546                 this.el.on("keydown", this.prepareEvent,  this);
14547                 this.el.on("keypress", this.relay,  this);
14548             }
14549                     this.disabled = false;
14550                 }
14551         },
14552
14553         /**
14554          * Disable this KeyNav
14555          */
14556         disable: function(){
14557                 if(!this.disabled){
14558                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14559                 this.el.un("keydown", this.relay);
14560             }else{
14561                 this.el.un("keydown", this.prepareEvent);
14562                 this.el.un("keypress", this.relay);
14563             }
14564                     this.disabled = true;
14565                 }
14566         }
14567 };/*
14568  * Based on:
14569  * Ext JS Library 1.1.1
14570  * Copyright(c) 2006-2007, Ext JS, LLC.
14571  *
14572  * Originally Released Under LGPL - original licence link has changed is not relivant.
14573  *
14574  * Fork - LGPL
14575  * <script type="text/javascript">
14576  */
14577
14578  
14579 /**
14580  * @class Roo.KeyMap
14581  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14582  * The constructor accepts the same config object as defined by {@link #addBinding}.
14583  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14584  * combination it will call the function with this signature (if the match is a multi-key
14585  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14586  * A KeyMap can also handle a string representation of keys.<br />
14587  * Usage:
14588  <pre><code>
14589 // map one key by key code
14590 var map = new Roo.KeyMap("my-element", {
14591     key: 13, // or Roo.EventObject.ENTER
14592     fn: myHandler,
14593     scope: myObject
14594 });
14595
14596 // map multiple keys to one action by string
14597 var map = new Roo.KeyMap("my-element", {
14598     key: "a\r\n\t",
14599     fn: myHandler,
14600     scope: myObject
14601 });
14602
14603 // map multiple keys to multiple actions by strings and array of codes
14604 var map = new Roo.KeyMap("my-element", [
14605     {
14606         key: [10,13],
14607         fn: function(){ alert("Return was pressed"); }
14608     }, {
14609         key: "abc",
14610         fn: function(){ alert('a, b or c was pressed'); }
14611     }, {
14612         key: "\t",
14613         ctrl:true,
14614         shift:true,
14615         fn: function(){ alert('Control + shift + tab was pressed.'); }
14616     }
14617 ]);
14618 </code></pre>
14619  * <b>Note: A KeyMap starts enabled</b>
14620  * @constructor
14621  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14622  * @param {Object} config The config (see {@link #addBinding})
14623  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14624  */
14625 Roo.KeyMap = function(el, config, eventName){
14626     this.el  = Roo.get(el);
14627     this.eventName = eventName || "keydown";
14628     this.bindings = [];
14629     if(config){
14630         this.addBinding(config);
14631     }
14632     this.enable();
14633 };
14634
14635 Roo.KeyMap.prototype = {
14636     /**
14637      * True to stop the event from bubbling and prevent the default browser action if the
14638      * key was handled by the KeyMap (defaults to false)
14639      * @type Boolean
14640      */
14641     stopEvent : false,
14642
14643     /**
14644      * Add a new binding to this KeyMap. The following config object properties are supported:
14645      * <pre>
14646 Property    Type             Description
14647 ----------  ---------------  ----------------------------------------------------------------------
14648 key         String/Array     A single keycode or an array of keycodes to handle
14649 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14650 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14651 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14652 fn          Function         The function to call when KeyMap finds the expected key combination
14653 scope       Object           The scope of the callback function
14654 </pre>
14655      *
14656      * Usage:
14657      * <pre><code>
14658 // Create a KeyMap
14659 var map = new Roo.KeyMap(document, {
14660     key: Roo.EventObject.ENTER,
14661     fn: handleKey,
14662     scope: this
14663 });
14664
14665 //Add a new binding to the existing KeyMap later
14666 map.addBinding({
14667     key: 'abc',
14668     shift: true,
14669     fn: handleKey,
14670     scope: this
14671 });
14672 </code></pre>
14673      * @param {Object/Array} config A single KeyMap config or an array of configs
14674      */
14675         addBinding : function(config){
14676         if(config instanceof Array){
14677             for(var i = 0, len = config.length; i < len; i++){
14678                 this.addBinding(config[i]);
14679             }
14680             return;
14681         }
14682         var keyCode = config.key,
14683             shift = config.shift, 
14684             ctrl = config.ctrl, 
14685             alt = config.alt,
14686             fn = config.fn,
14687             scope = config.scope;
14688         if(typeof keyCode == "string"){
14689             var ks = [];
14690             var keyString = keyCode.toUpperCase();
14691             for(var j = 0, len = keyString.length; j < len; j++){
14692                 ks.push(keyString.charCodeAt(j));
14693             }
14694             keyCode = ks;
14695         }
14696         var keyArray = keyCode instanceof Array;
14697         var handler = function(e){
14698             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14699                 var k = e.getKey();
14700                 if(keyArray){
14701                     for(var i = 0, len = keyCode.length; i < len; i++){
14702                         if(keyCode[i] == k){
14703                           if(this.stopEvent){
14704                               e.stopEvent();
14705                           }
14706                           fn.call(scope || window, k, e);
14707                           return;
14708                         }
14709                     }
14710                 }else{
14711                     if(k == keyCode){
14712                         if(this.stopEvent){
14713                            e.stopEvent();
14714                         }
14715                         fn.call(scope || window, k, e);
14716                     }
14717                 }
14718             }
14719         };
14720         this.bindings.push(handler);  
14721         },
14722
14723     /**
14724      * Shorthand for adding a single key listener
14725      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14726      * following options:
14727      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14728      * @param {Function} fn The function to call
14729      * @param {Object} scope (optional) The scope of the function
14730      */
14731     on : function(key, fn, scope){
14732         var keyCode, shift, ctrl, alt;
14733         if(typeof key == "object" && !(key instanceof Array)){
14734             keyCode = key.key;
14735             shift = key.shift;
14736             ctrl = key.ctrl;
14737             alt = key.alt;
14738         }else{
14739             keyCode = key;
14740         }
14741         this.addBinding({
14742             key: keyCode,
14743             shift: shift,
14744             ctrl: ctrl,
14745             alt: alt,
14746             fn: fn,
14747             scope: scope
14748         })
14749     },
14750
14751     // private
14752     handleKeyDown : function(e){
14753             if(this.enabled){ //just in case
14754             var b = this.bindings;
14755             for(var i = 0, len = b.length; i < len; i++){
14756                 b[i].call(this, e);
14757             }
14758             }
14759         },
14760         
14761         /**
14762          * Returns true if this KeyMap is enabled
14763          * @return {Boolean} 
14764          */
14765         isEnabled : function(){
14766             return this.enabled;  
14767         },
14768         
14769         /**
14770          * Enables this KeyMap
14771          */
14772         enable: function(){
14773                 if(!this.enabled){
14774                     this.el.on(this.eventName, this.handleKeyDown, this);
14775                     this.enabled = true;
14776                 }
14777         },
14778
14779         /**
14780          * Disable this KeyMap
14781          */
14782         disable: function(){
14783                 if(this.enabled){
14784                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14785                     this.enabled = false;
14786                 }
14787         }
14788 };/*
14789  * Based on:
14790  * Ext JS Library 1.1.1
14791  * Copyright(c) 2006-2007, Ext JS, LLC.
14792  *
14793  * Originally Released Under LGPL - original licence link has changed is not relivant.
14794  *
14795  * Fork - LGPL
14796  * <script type="text/javascript">
14797  */
14798
14799  
14800 /**
14801  * @class Roo.util.TextMetrics
14802  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14803  * wide, in pixels, a given block of text will be.
14804  * @singleton
14805  */
14806 Roo.util.TextMetrics = function(){
14807     var shared;
14808     return {
14809         /**
14810          * Measures the size of the specified text
14811          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14812          * that can affect the size of the rendered text
14813          * @param {String} text The text to measure
14814          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14815          * in order to accurately measure the text height
14816          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14817          */
14818         measure : function(el, text, fixedWidth){
14819             if(!shared){
14820                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14821             }
14822             shared.bind(el);
14823             shared.setFixedWidth(fixedWidth || 'auto');
14824             return shared.getSize(text);
14825         },
14826
14827         /**
14828          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14829          * the overhead of multiple calls to initialize the style properties on each measurement.
14830          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14831          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14832          * in order to accurately measure the text height
14833          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14834          */
14835         createInstance : function(el, fixedWidth){
14836             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14837         }
14838     };
14839 }();
14840
14841  
14842
14843 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14844     var ml = new Roo.Element(document.createElement('div'));
14845     document.body.appendChild(ml.dom);
14846     ml.position('absolute');
14847     ml.setLeftTop(-1000, -1000);
14848     ml.hide();
14849
14850     if(fixedWidth){
14851         ml.setWidth(fixedWidth);
14852     }
14853      
14854     var instance = {
14855         /**
14856          * Returns the size of the specified text based on the internal element's style and width properties
14857          * @memberOf Roo.util.TextMetrics.Instance#
14858          * @param {String} text The text to measure
14859          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14860          */
14861         getSize : function(text){
14862             ml.update(text);
14863             var s = ml.getSize();
14864             ml.update('');
14865             return s;
14866         },
14867
14868         /**
14869          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14870          * that can affect the size of the rendered text
14871          * @memberOf Roo.util.TextMetrics.Instance#
14872          * @param {String/HTMLElement} el The element, dom node or id
14873          */
14874         bind : function(el){
14875             ml.setStyle(
14876                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14877             );
14878         },
14879
14880         /**
14881          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14882          * to set a fixed width in order to accurately measure the text height.
14883          * @memberOf Roo.util.TextMetrics.Instance#
14884          * @param {Number} width The width to set on the element
14885          */
14886         setFixedWidth : function(width){
14887             ml.setWidth(width);
14888         },
14889
14890         /**
14891          * Returns the measured width of the specified text
14892          * @memberOf Roo.util.TextMetrics.Instance#
14893          * @param {String} text The text to measure
14894          * @return {Number} width The width in pixels
14895          */
14896         getWidth : function(text){
14897             ml.dom.style.width = 'auto';
14898             return this.getSize(text).width;
14899         },
14900
14901         /**
14902          * Returns the measured height of the specified text.  For multiline text, be sure to call
14903          * {@link #setFixedWidth} if necessary.
14904          * @memberOf Roo.util.TextMetrics.Instance#
14905          * @param {String} text The text to measure
14906          * @return {Number} height The height in pixels
14907          */
14908         getHeight : function(text){
14909             return this.getSize(text).height;
14910         }
14911     };
14912
14913     instance.bind(bindTo);
14914
14915     return instance;
14916 };
14917
14918 // backwards compat
14919 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14920  * Based on:
14921  * Ext JS Library 1.1.1
14922  * Copyright(c) 2006-2007, Ext JS, LLC.
14923  *
14924  * Originally Released Under LGPL - original licence link has changed is not relivant.
14925  *
14926  * Fork - LGPL
14927  * <script type="text/javascript">
14928  */
14929
14930 /**
14931  * @class Roo.state.Provider
14932  * Abstract base class for state provider implementations. This class provides methods
14933  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14934  * Provider interface.
14935  */
14936 Roo.state.Provider = function(){
14937     /**
14938      * @event statechange
14939      * Fires when a state change occurs.
14940      * @param {Provider} this This state provider
14941      * @param {String} key The state key which was changed
14942      * @param {String} value The encoded value for the state
14943      */
14944     this.addEvents({
14945         "statechange": true
14946     });
14947     this.state = {};
14948     Roo.state.Provider.superclass.constructor.call(this);
14949 };
14950 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14951     /**
14952      * Returns the current value for a key
14953      * @param {String} name The key name
14954      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14955      * @return {Mixed} The state data
14956      */
14957     get : function(name, defaultValue){
14958         return typeof this.state[name] == "undefined" ?
14959             defaultValue : this.state[name];
14960     },
14961     
14962     /**
14963      * Clears a value from the state
14964      * @param {String} name The key name
14965      */
14966     clear : function(name){
14967         delete this.state[name];
14968         this.fireEvent("statechange", this, name, null);
14969     },
14970     
14971     /**
14972      * Sets the value for a key
14973      * @param {String} name The key name
14974      * @param {Mixed} value The value to set
14975      */
14976     set : function(name, value){
14977         this.state[name] = value;
14978         this.fireEvent("statechange", this, name, value);
14979     },
14980     
14981     /**
14982      * Decodes a string previously encoded with {@link #encodeValue}.
14983      * @param {String} value The value to decode
14984      * @return {Mixed} The decoded value
14985      */
14986     decodeValue : function(cookie){
14987         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14988         var matches = re.exec(unescape(cookie));
14989         if(!matches || !matches[1]) {
14990             return; // non state cookie
14991         }
14992         var type = matches[1];
14993         var v = matches[2];
14994         switch(type){
14995             case "n":
14996                 return parseFloat(v);
14997             case "d":
14998                 return new Date(Date.parse(v));
14999             case "b":
15000                 return (v == "1");
15001             case "a":
15002                 var all = [];
15003                 var values = v.split("^");
15004                 for(var i = 0, len = values.length; i < len; i++){
15005                     all.push(this.decodeValue(values[i]));
15006                 }
15007                 return all;
15008            case "o":
15009                 var all = {};
15010                 var values = v.split("^");
15011                 for(var i = 0, len = values.length; i < len; i++){
15012                     var kv = values[i].split("=");
15013                     all[kv[0]] = this.decodeValue(kv[1]);
15014                 }
15015                 return all;
15016            default:
15017                 return v;
15018         }
15019     },
15020     
15021     /**
15022      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15023      * @param {Mixed} value The value to encode
15024      * @return {String} The encoded value
15025      */
15026     encodeValue : function(v){
15027         var enc;
15028         if(typeof v == "number"){
15029             enc = "n:" + v;
15030         }else if(typeof v == "boolean"){
15031             enc = "b:" + (v ? "1" : "0");
15032         }else if(v instanceof Date){
15033             enc = "d:" + v.toGMTString();
15034         }else if(v instanceof Array){
15035             var flat = "";
15036             for(var i = 0, len = v.length; i < len; i++){
15037                 flat += this.encodeValue(v[i]);
15038                 if(i != len-1) {
15039                     flat += "^";
15040                 }
15041             }
15042             enc = "a:" + flat;
15043         }else if(typeof v == "object"){
15044             var flat = "";
15045             for(var key in v){
15046                 if(typeof v[key] != "function"){
15047                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15048                 }
15049             }
15050             enc = "o:" + flat.substring(0, flat.length-1);
15051         }else{
15052             enc = "s:" + v;
15053         }
15054         return escape(enc);        
15055     }
15056 });
15057
15058 /*
15059  * Based on:
15060  * Ext JS Library 1.1.1
15061  * Copyright(c) 2006-2007, Ext JS, LLC.
15062  *
15063  * Originally Released Under LGPL - original licence link has changed is not relivant.
15064  *
15065  * Fork - LGPL
15066  * <script type="text/javascript">
15067  */
15068 /**
15069  * @class Roo.state.Manager
15070  * This is the global state manager. By default all components that are "state aware" check this class
15071  * for state information if you don't pass them a custom state provider. In order for this class
15072  * to be useful, it must be initialized with a provider when your application initializes.
15073  <pre><code>
15074 // in your initialization function
15075 init : function(){
15076    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15077    ...
15078    // supposed you have a {@link Roo.BorderLayout}
15079    var layout = new Roo.BorderLayout(...);
15080    layout.restoreState();
15081    // or a {Roo.BasicDialog}
15082    var dialog = new Roo.BasicDialog(...);
15083    dialog.restoreState();
15084  </code></pre>
15085  * @singleton
15086  */
15087 Roo.state.Manager = function(){
15088     var provider = new Roo.state.Provider();
15089     
15090     return {
15091         /**
15092          * Configures the default state provider for your application
15093          * @param {Provider} stateProvider The state provider to set
15094          */
15095         setProvider : function(stateProvider){
15096             provider = stateProvider;
15097         },
15098         
15099         /**
15100          * Returns the current value for a key
15101          * @param {String} name The key name
15102          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15103          * @return {Mixed} The state data
15104          */
15105         get : function(key, defaultValue){
15106             return provider.get(key, defaultValue);
15107         },
15108         
15109         /**
15110          * Sets the value for a key
15111          * @param {String} name The key name
15112          * @param {Mixed} value The state data
15113          */
15114          set : function(key, value){
15115             provider.set(key, value);
15116         },
15117         
15118         /**
15119          * Clears a value from the state
15120          * @param {String} name The key name
15121          */
15122         clear : function(key){
15123             provider.clear(key);
15124         },
15125         
15126         /**
15127          * Gets the currently configured state provider
15128          * @return {Provider} The state provider
15129          */
15130         getProvider : function(){
15131             return provider;
15132         }
15133     };
15134 }();
15135 /*
15136  * Based on:
15137  * Ext JS Library 1.1.1
15138  * Copyright(c) 2006-2007, Ext JS, LLC.
15139  *
15140  * Originally Released Under LGPL - original licence link has changed is not relivant.
15141  *
15142  * Fork - LGPL
15143  * <script type="text/javascript">
15144  */
15145 /**
15146  * @class Roo.state.CookieProvider
15147  * @extends Roo.state.Provider
15148  * The default Provider implementation which saves state via cookies.
15149  * <br />Usage:
15150  <pre><code>
15151    var cp = new Roo.state.CookieProvider({
15152        path: "/cgi-bin/",
15153        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15154        domain: "roojs.com"
15155    })
15156    Roo.state.Manager.setProvider(cp);
15157  </code></pre>
15158  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15159  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15160  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15161  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15162  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15163  * domain the page is running on including the 'www' like 'www.roojs.com')
15164  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15165  * @constructor
15166  * Create a new CookieProvider
15167  * @param {Object} config The configuration object
15168  */
15169 Roo.state.CookieProvider = function(config){
15170     Roo.state.CookieProvider.superclass.constructor.call(this);
15171     this.path = "/";
15172     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15173     this.domain = null;
15174     this.secure = false;
15175     Roo.apply(this, config);
15176     this.state = this.readCookies();
15177 };
15178
15179 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15180     // private
15181     set : function(name, value){
15182         if(typeof value == "undefined" || value === null){
15183             this.clear(name);
15184             return;
15185         }
15186         this.setCookie(name, value);
15187         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15188     },
15189
15190     // private
15191     clear : function(name){
15192         this.clearCookie(name);
15193         Roo.state.CookieProvider.superclass.clear.call(this, name);
15194     },
15195
15196     // private
15197     readCookies : function(){
15198         var cookies = {};
15199         var c = document.cookie + ";";
15200         var re = /\s?(.*?)=(.*?);/g;
15201         var matches;
15202         while((matches = re.exec(c)) != null){
15203             var name = matches[1];
15204             var value = matches[2];
15205             if(name && name.substring(0,3) == "ys-"){
15206                 cookies[name.substr(3)] = this.decodeValue(value);
15207             }
15208         }
15209         return cookies;
15210     },
15211
15212     // private
15213     setCookie : function(name, value){
15214         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15215            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15216            ((this.path == null) ? "" : ("; path=" + this.path)) +
15217            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15218            ((this.secure == true) ? "; secure" : "");
15219     },
15220
15221     // private
15222     clearCookie : function(name){
15223         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15224            ((this.path == null) ? "" : ("; path=" + this.path)) +
15225            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15226            ((this.secure == true) ? "; secure" : "");
15227     }
15228 });/*
15229  * Based on:
15230  * Ext JS Library 1.1.1
15231  * Copyright(c) 2006-2007, Ext JS, LLC.
15232  *
15233  * Originally Released Under LGPL - original licence link has changed is not relivant.
15234  *
15235  * Fork - LGPL
15236  * <script type="text/javascript">
15237  */
15238  
15239
15240 /**
15241  * @class Roo.ComponentMgr
15242  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15243  * @singleton
15244  */
15245 Roo.ComponentMgr = function(){
15246     var all = new Roo.util.MixedCollection();
15247
15248     return {
15249         /**
15250          * Registers a component.
15251          * @param {Roo.Component} c The component
15252          */
15253         register : function(c){
15254             all.add(c);
15255         },
15256
15257         /**
15258          * Unregisters a component.
15259          * @param {Roo.Component} c The component
15260          */
15261         unregister : function(c){
15262             all.remove(c);
15263         },
15264
15265         /**
15266          * Returns a component by id
15267          * @param {String} id The component id
15268          */
15269         get : function(id){
15270             return all.get(id);
15271         },
15272
15273         /**
15274          * Registers a function that will be called when a specified component is added to ComponentMgr
15275          * @param {String} id The component id
15276          * @param {Funtction} fn The callback function
15277          * @param {Object} scope The scope of the callback
15278          */
15279         onAvailable : function(id, fn, scope){
15280             all.on("add", function(index, o){
15281                 if(o.id == id){
15282                     fn.call(scope || o, o);
15283                     all.un("add", fn, scope);
15284                 }
15285             });
15286         }
15287     };
15288 }();/*
15289  * Based on:
15290  * Ext JS Library 1.1.1
15291  * Copyright(c) 2006-2007, Ext JS, LLC.
15292  *
15293  * Originally Released Under LGPL - original licence link has changed is not relivant.
15294  *
15295  * Fork - LGPL
15296  * <script type="text/javascript">
15297  */
15298  
15299 /**
15300  * @class Roo.Component
15301  * @extends Roo.util.Observable
15302  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15303  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15304  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15305  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15306  * All visual components (widgets) that require rendering into a layout should subclass Component.
15307  * @constructor
15308  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15309  * 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
15310  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15311  */
15312 Roo.Component = function(config){
15313     config = config || {};
15314     if(config.tagName || config.dom || typeof config == "string"){ // element object
15315         config = {el: config, id: config.id || config};
15316     }
15317     this.initialConfig = config;
15318
15319     Roo.apply(this, config);
15320     this.addEvents({
15321         /**
15322          * @event disable
15323          * Fires after the component is disabled.
15324              * @param {Roo.Component} this
15325              */
15326         disable : true,
15327         /**
15328          * @event enable
15329          * Fires after the component is enabled.
15330              * @param {Roo.Component} this
15331              */
15332         enable : true,
15333         /**
15334          * @event beforeshow
15335          * Fires before the component is shown.  Return false to stop the show.
15336              * @param {Roo.Component} this
15337              */
15338         beforeshow : true,
15339         /**
15340          * @event show
15341          * Fires after the component is shown.
15342              * @param {Roo.Component} this
15343              */
15344         show : true,
15345         /**
15346          * @event beforehide
15347          * Fires before the component is hidden. Return false to stop the hide.
15348              * @param {Roo.Component} this
15349              */
15350         beforehide : true,
15351         /**
15352          * @event hide
15353          * Fires after the component is hidden.
15354              * @param {Roo.Component} this
15355              */
15356         hide : true,
15357         /**
15358          * @event beforerender
15359          * Fires before the component is rendered. Return false to stop the render.
15360              * @param {Roo.Component} this
15361              */
15362         beforerender : true,
15363         /**
15364          * @event render
15365          * Fires after the component is rendered.
15366              * @param {Roo.Component} this
15367              */
15368         render : true,
15369         /**
15370          * @event beforedestroy
15371          * Fires before the component is destroyed. Return false to stop the destroy.
15372              * @param {Roo.Component} this
15373              */
15374         beforedestroy : true,
15375         /**
15376          * @event destroy
15377          * Fires after the component is destroyed.
15378              * @param {Roo.Component} this
15379              */
15380         destroy : true
15381     });
15382     if(!this.id){
15383         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15384     }
15385     Roo.ComponentMgr.register(this);
15386     Roo.Component.superclass.constructor.call(this);
15387     this.initComponent();
15388     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15389         this.render(this.renderTo);
15390         delete this.renderTo;
15391     }
15392 };
15393
15394 /** @private */
15395 Roo.Component.AUTO_ID = 1000;
15396
15397 Roo.extend(Roo.Component, Roo.util.Observable, {
15398     /**
15399      * @scope Roo.Component.prototype
15400      * @type {Boolean}
15401      * true if this component is hidden. Read-only.
15402      */
15403     hidden : false,
15404     /**
15405      * @type {Boolean}
15406      * true if this component is disabled. Read-only.
15407      */
15408     disabled : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component has been rendered. Read-only.
15412      */
15413     rendered : false,
15414     
15415     /** @cfg {String} disableClass
15416      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15417      */
15418     disabledClass : "x-item-disabled",
15419         /** @cfg {Boolean} allowDomMove
15420          * Whether the component can move the Dom node when rendering (defaults to true).
15421          */
15422     allowDomMove : true,
15423     /** @cfg {String} hideMode (display|visibility)
15424      * How this component should hidden. Supported values are
15425      * "visibility" (css visibility), "offsets" (negative offset position) and
15426      * "display" (css display) - defaults to "display".
15427      */
15428     hideMode: 'display',
15429
15430     /** @private */
15431     ctype : "Roo.Component",
15432
15433     /**
15434      * @cfg {String} actionMode 
15435      * which property holds the element that used for  hide() / show() / disable() / enable()
15436      * default is 'el' 
15437      */
15438     actionMode : "el",
15439
15440     /** @private */
15441     getActionEl : function(){
15442         return this[this.actionMode];
15443     },
15444
15445     initComponent : Roo.emptyFn,
15446     /**
15447      * If this is a lazy rendering component, render it to its container element.
15448      * @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.
15449      */
15450     render : function(container, position){
15451         
15452         if(this.rendered){
15453             return this;
15454         }
15455         
15456         if(this.fireEvent("beforerender", this) === false){
15457             return false;
15458         }
15459         
15460         if(!container && this.el){
15461             this.el = Roo.get(this.el);
15462             container = this.el.dom.parentNode;
15463             this.allowDomMove = false;
15464         }
15465         this.container = Roo.get(container);
15466         this.rendered = true;
15467         if(position !== undefined){
15468             if(typeof position == 'number'){
15469                 position = this.container.dom.childNodes[position];
15470             }else{
15471                 position = Roo.getDom(position);
15472             }
15473         }
15474         this.onRender(this.container, position || null);
15475         if(this.cls){
15476             this.el.addClass(this.cls);
15477             delete this.cls;
15478         }
15479         if(this.style){
15480             this.el.applyStyles(this.style);
15481             delete this.style;
15482         }
15483         this.fireEvent("render", this);
15484         this.afterRender(this.container);
15485         if(this.hidden){
15486             this.hide();
15487         }
15488         if(this.disabled){
15489             this.disable();
15490         }
15491
15492         return this;
15493         
15494     },
15495
15496     /** @private */
15497     // default function is not really useful
15498     onRender : function(ct, position){
15499         if(this.el){
15500             this.el = Roo.get(this.el);
15501             if(this.allowDomMove !== false){
15502                 ct.dom.insertBefore(this.el.dom, position);
15503             }
15504         }
15505     },
15506
15507     /** @private */
15508     getAutoCreate : function(){
15509         var cfg = typeof this.autoCreate == "object" ?
15510                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15511         if(this.id && !cfg.id){
15512             cfg.id = this.id;
15513         }
15514         return cfg;
15515     },
15516
15517     /** @private */
15518     afterRender : Roo.emptyFn,
15519
15520     /**
15521      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15522      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15523      */
15524     destroy : function(){
15525         if(this.fireEvent("beforedestroy", this) !== false){
15526             this.purgeListeners();
15527             this.beforeDestroy();
15528             if(this.rendered){
15529                 this.el.removeAllListeners();
15530                 this.el.remove();
15531                 if(this.actionMode == "container"){
15532                     this.container.remove();
15533                 }
15534             }
15535             this.onDestroy();
15536             Roo.ComponentMgr.unregister(this);
15537             this.fireEvent("destroy", this);
15538         }
15539     },
15540
15541         /** @private */
15542     beforeDestroy : function(){
15543
15544     },
15545
15546         /** @private */
15547         onDestroy : function(){
15548
15549     },
15550
15551     /**
15552      * Returns the underlying {@link Roo.Element}.
15553      * @return {Roo.Element} The element
15554      */
15555     getEl : function(){
15556         return this.el;
15557     },
15558
15559     /**
15560      * Returns the id of this component.
15561      * @return {String}
15562      */
15563     getId : function(){
15564         return this.id;
15565     },
15566
15567     /**
15568      * Try to focus this component.
15569      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15570      * @return {Roo.Component} this
15571      */
15572     focus : function(selectText){
15573         if(this.rendered){
15574             this.el.focus();
15575             if(selectText === true){
15576                 this.el.dom.select();
15577             }
15578         }
15579         return this;
15580     },
15581
15582     /** @private */
15583     blur : function(){
15584         if(this.rendered){
15585             this.el.blur();
15586         }
15587         return this;
15588     },
15589
15590     /**
15591      * Disable this component.
15592      * @return {Roo.Component} this
15593      */
15594     disable : function(){
15595         if(this.rendered){
15596             this.onDisable();
15597         }
15598         this.disabled = true;
15599         this.fireEvent("disable", this);
15600         return this;
15601     },
15602
15603         // private
15604     onDisable : function(){
15605         this.getActionEl().addClass(this.disabledClass);
15606         this.el.dom.disabled = true;
15607     },
15608
15609     /**
15610      * Enable this component.
15611      * @return {Roo.Component} this
15612      */
15613     enable : function(){
15614         if(this.rendered){
15615             this.onEnable();
15616         }
15617         this.disabled = false;
15618         this.fireEvent("enable", this);
15619         return this;
15620     },
15621
15622         // private
15623     onEnable : function(){
15624         this.getActionEl().removeClass(this.disabledClass);
15625         this.el.dom.disabled = false;
15626     },
15627
15628     /**
15629      * Convenience function for setting disabled/enabled by boolean.
15630      * @param {Boolean} disabled
15631      */
15632     setDisabled : function(disabled){
15633         this[disabled ? "disable" : "enable"]();
15634     },
15635
15636     /**
15637      * Show this component.
15638      * @return {Roo.Component} this
15639      */
15640     show: function(){
15641         if(this.fireEvent("beforeshow", this) !== false){
15642             this.hidden = false;
15643             if(this.rendered){
15644                 this.onShow();
15645             }
15646             this.fireEvent("show", this);
15647         }
15648         return this;
15649     },
15650
15651     // private
15652     onShow : function(){
15653         var ae = this.getActionEl();
15654         if(this.hideMode == 'visibility'){
15655             ae.dom.style.visibility = "visible";
15656         }else if(this.hideMode == 'offsets'){
15657             ae.removeClass('x-hidden');
15658         }else{
15659             ae.dom.style.display = "";
15660         }
15661     },
15662
15663     /**
15664      * Hide this component.
15665      * @return {Roo.Component} this
15666      */
15667     hide: function(){
15668         if(this.fireEvent("beforehide", this) !== false){
15669             this.hidden = true;
15670             if(this.rendered){
15671                 this.onHide();
15672             }
15673             this.fireEvent("hide", this);
15674         }
15675         return this;
15676     },
15677
15678     // private
15679     onHide : function(){
15680         var ae = this.getActionEl();
15681         if(this.hideMode == 'visibility'){
15682             ae.dom.style.visibility = "hidden";
15683         }else if(this.hideMode == 'offsets'){
15684             ae.addClass('x-hidden');
15685         }else{
15686             ae.dom.style.display = "none";
15687         }
15688     },
15689
15690     /**
15691      * Convenience function to hide or show this component by boolean.
15692      * @param {Boolean} visible True to show, false to hide
15693      * @return {Roo.Component} this
15694      */
15695     setVisible: function(visible){
15696         if(visible) {
15697             this.show();
15698         }else{
15699             this.hide();
15700         }
15701         return this;
15702     },
15703
15704     /**
15705      * Returns true if this component is visible.
15706      */
15707     isVisible : function(){
15708         return this.getActionEl().isVisible();
15709     },
15710
15711     cloneConfig : function(overrides){
15712         overrides = overrides || {};
15713         var id = overrides.id || Roo.id();
15714         var cfg = Roo.applyIf(overrides, this.initialConfig);
15715         cfg.id = id; // prevent dup id
15716         return new this.constructor(cfg);
15717     }
15718 });/*
15719  * Based on:
15720  * Ext JS Library 1.1.1
15721  * Copyright(c) 2006-2007, Ext JS, LLC.
15722  *
15723  * Originally Released Under LGPL - original licence link has changed is not relivant.
15724  *
15725  * Fork - LGPL
15726  * <script type="text/javascript">
15727  */
15728
15729 /**
15730  * @class Roo.BoxComponent
15731  * @extends Roo.Component
15732  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15733  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15734  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15735  * layout containers.
15736  * @constructor
15737  * @param {Roo.Element/String/Object} config The configuration options.
15738  */
15739 Roo.BoxComponent = function(config){
15740     Roo.Component.call(this, config);
15741     this.addEvents({
15742         /**
15743          * @event resize
15744          * Fires after the component is resized.
15745              * @param {Roo.Component} this
15746              * @param {Number} adjWidth The box-adjusted width that was set
15747              * @param {Number} adjHeight The box-adjusted height that was set
15748              * @param {Number} rawWidth The width that was originally specified
15749              * @param {Number} rawHeight The height that was originally specified
15750              */
15751         resize : true,
15752         /**
15753          * @event move
15754          * Fires after the component is moved.
15755              * @param {Roo.Component} this
15756              * @param {Number} x The new x position
15757              * @param {Number} y The new y position
15758              */
15759         move : true
15760     });
15761 };
15762
15763 Roo.extend(Roo.BoxComponent, Roo.Component, {
15764     // private, set in afterRender to signify that the component has been rendered
15765     boxReady : false,
15766     // private, used to defer height settings to subclasses
15767     deferHeight: false,
15768     /** @cfg {Number} width
15769      * width (optional) size of component
15770      */
15771      /** @cfg {Number} height
15772      * height (optional) size of component
15773      */
15774      
15775     /**
15776      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15777      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15778      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15779      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15780      * @return {Roo.BoxComponent} this
15781      */
15782     setSize : function(w, h){
15783         // support for standard size objects
15784         if(typeof w == 'object'){
15785             h = w.height;
15786             w = w.width;
15787         }
15788         // not rendered
15789         if(!this.boxReady){
15790             this.width = w;
15791             this.height = h;
15792             return this;
15793         }
15794
15795         // prevent recalcs when not needed
15796         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15797             return this;
15798         }
15799         this.lastSize = {width: w, height: h};
15800
15801         var adj = this.adjustSize(w, h);
15802         var aw = adj.width, ah = adj.height;
15803         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15804             var rz = this.getResizeEl();
15805             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15806                 rz.setSize(aw, ah);
15807             }else if(!this.deferHeight && ah !== undefined){
15808                 rz.setHeight(ah);
15809             }else if(aw !== undefined){
15810                 rz.setWidth(aw);
15811             }
15812             this.onResize(aw, ah, w, h);
15813             this.fireEvent('resize', this, aw, ah, w, h);
15814         }
15815         return this;
15816     },
15817
15818     /**
15819      * Gets the current size of the component's underlying element.
15820      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15821      */
15822     getSize : function(){
15823         return this.el.getSize();
15824     },
15825
15826     /**
15827      * Gets the current XY position of the component's underlying element.
15828      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15829      * @return {Array} The XY position of the element (e.g., [100, 200])
15830      */
15831     getPosition : function(local){
15832         if(local === true){
15833             return [this.el.getLeft(true), this.el.getTop(true)];
15834         }
15835         return this.xy || this.el.getXY();
15836     },
15837
15838     /**
15839      * Gets the current box measurements of the component's underlying element.
15840      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15841      * @returns {Object} box An object in the format {x, y, width, height}
15842      */
15843     getBox : function(local){
15844         var s = this.el.getSize();
15845         if(local){
15846             s.x = this.el.getLeft(true);
15847             s.y = this.el.getTop(true);
15848         }else{
15849             var xy = this.xy || this.el.getXY();
15850             s.x = xy[0];
15851             s.y = xy[1];
15852         }
15853         return s;
15854     },
15855
15856     /**
15857      * Sets the current box measurements of the component's underlying element.
15858      * @param {Object} box An object in the format {x, y, width, height}
15859      * @returns {Roo.BoxComponent} this
15860      */
15861     updateBox : function(box){
15862         this.setSize(box.width, box.height);
15863         this.setPagePosition(box.x, box.y);
15864         return this;
15865     },
15866
15867     // protected
15868     getResizeEl : function(){
15869         return this.resizeEl || this.el;
15870     },
15871
15872     // protected
15873     getPositionEl : function(){
15874         return this.positionEl || this.el;
15875     },
15876
15877     /**
15878      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15879      * This method fires the move event.
15880      * @param {Number} left The new left
15881      * @param {Number} top The new top
15882      * @returns {Roo.BoxComponent} this
15883      */
15884     setPosition : function(x, y){
15885         this.x = x;
15886         this.y = y;
15887         if(!this.boxReady){
15888             return this;
15889         }
15890         var adj = this.adjustPosition(x, y);
15891         var ax = adj.x, ay = adj.y;
15892
15893         var el = this.getPositionEl();
15894         if(ax !== undefined || ay !== undefined){
15895             if(ax !== undefined && ay !== undefined){
15896                 el.setLeftTop(ax, ay);
15897             }else if(ax !== undefined){
15898                 el.setLeft(ax);
15899             }else if(ay !== undefined){
15900                 el.setTop(ay);
15901             }
15902             this.onPosition(ax, ay);
15903             this.fireEvent('move', this, ax, ay);
15904         }
15905         return this;
15906     },
15907
15908     /**
15909      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15910      * This method fires the move event.
15911      * @param {Number} x The new x position
15912      * @param {Number} y The new y position
15913      * @returns {Roo.BoxComponent} this
15914      */
15915     setPagePosition : function(x, y){
15916         this.pageX = x;
15917         this.pageY = y;
15918         if(!this.boxReady){
15919             return;
15920         }
15921         if(x === undefined || y === undefined){ // cannot translate undefined points
15922             return;
15923         }
15924         var p = this.el.translatePoints(x, y);
15925         this.setPosition(p.left, p.top);
15926         return this;
15927     },
15928
15929     // private
15930     onRender : function(ct, position){
15931         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15932         if(this.resizeEl){
15933             this.resizeEl = Roo.get(this.resizeEl);
15934         }
15935         if(this.positionEl){
15936             this.positionEl = Roo.get(this.positionEl);
15937         }
15938     },
15939
15940     // private
15941     afterRender : function(){
15942         Roo.BoxComponent.superclass.afterRender.call(this);
15943         this.boxReady = true;
15944         this.setSize(this.width, this.height);
15945         if(this.x || this.y){
15946             this.setPosition(this.x, this.y);
15947         }
15948         if(this.pageX || this.pageY){
15949             this.setPagePosition(this.pageX, this.pageY);
15950         }
15951     },
15952
15953     /**
15954      * Force the component's size to recalculate based on the underlying element's current height and width.
15955      * @returns {Roo.BoxComponent} this
15956      */
15957     syncSize : function(){
15958         delete this.lastSize;
15959         this.setSize(this.el.getWidth(), this.el.getHeight());
15960         return this;
15961     },
15962
15963     /**
15964      * Called after the component is resized, this method is empty by default but can be implemented by any
15965      * subclass that needs to perform custom logic after a resize occurs.
15966      * @param {Number} adjWidth The box-adjusted width that was set
15967      * @param {Number} adjHeight The box-adjusted height that was set
15968      * @param {Number} rawWidth The width that was originally specified
15969      * @param {Number} rawHeight The height that was originally specified
15970      */
15971     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15972
15973     },
15974
15975     /**
15976      * Called after the component is moved, this method is empty by default but can be implemented by any
15977      * subclass that needs to perform custom logic after a move occurs.
15978      * @param {Number} x The new x position
15979      * @param {Number} y The new y position
15980      */
15981     onPosition : function(x, y){
15982
15983     },
15984
15985     // private
15986     adjustSize : function(w, h){
15987         if(this.autoWidth){
15988             w = 'auto';
15989         }
15990         if(this.autoHeight){
15991             h = 'auto';
15992         }
15993         return {width : w, height: h};
15994     },
15995
15996     // private
15997     adjustPosition : function(x, y){
15998         return {x : x, y: y};
15999     }
16000 });/*
16001  * Original code for Roojs - LGPL
16002  * <script type="text/javascript">
16003  */
16004  
16005 /**
16006  * @class Roo.XComponent
16007  * A delayed Element creator...
16008  * Or a way to group chunks of interface together.
16009  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16010  *  used in conjunction with XComponent.build() it will create an instance of each element,
16011  *  then call addxtype() to build the User interface.
16012  * 
16013  * Mypart.xyx = new Roo.XComponent({
16014
16015     parent : 'Mypart.xyz', // empty == document.element.!!
16016     order : '001',
16017     name : 'xxxx'
16018     region : 'xxxx'
16019     disabled : function() {} 
16020      
16021     tree : function() { // return an tree of xtype declared components
16022         var MODULE = this;
16023         return 
16024         {
16025             xtype : 'NestedLayoutPanel',
16026             // technicall
16027         }
16028      ]
16029  *})
16030  *
16031  *
16032  * It can be used to build a big heiracy, with parent etc.
16033  * or you can just use this to render a single compoent to a dom element
16034  * MYPART.render(Roo.Element | String(id) | dom_element )
16035  *
16036  *
16037  * Usage patterns.
16038  *
16039  * Classic Roo
16040  *
16041  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16042  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16043  *
16044  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16045  *
16046  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16047  * - if mulitple topModules exist, the last one is defined as the top module.
16048  *
16049  * Embeded Roo
16050  * 
16051  * When the top level or multiple modules are to embedded into a existing HTML page,
16052  * the parent element can container '#id' of the element where the module will be drawn.
16053  *
16054  * Bootstrap Roo
16055  *
16056  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16057  * it relies more on a include mechanism, where sub modules are included into an outer page.
16058  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16059  * 
16060  * Bootstrap Roo Included elements
16061  *
16062  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16063  * hence confusing the component builder as it thinks there are multiple top level elements. 
16064  *
16065  * String Over-ride & Translations
16066  *
16067  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16068  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16069  * are needed. @see Roo.XComponent.overlayString  
16070  * 
16071  * 
16072  * 
16073  * @extends Roo.util.Observable
16074  * @constructor
16075  * @param cfg {Object} configuration of component
16076  * 
16077  */
16078 Roo.XComponent = function(cfg) {
16079     Roo.apply(this, cfg);
16080     this.addEvents({ 
16081         /**
16082              * @event built
16083              * Fires when this the componnt is built
16084              * @param {Roo.XComponent} c the component
16085              */
16086         'built' : true
16087         
16088     });
16089     this.region = this.region || 'center'; // default..
16090     Roo.XComponent.register(this);
16091     this.modules = false;
16092     this.el = false; // where the layout goes..
16093     
16094     
16095 }
16096 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16097     /**
16098      * @property el
16099      * The created element (with Roo.factory())
16100      * @type {Roo.Layout}
16101      */
16102     el  : false,
16103     
16104     /**
16105      * @property el
16106      * for BC  - use el in new code
16107      * @type {Roo.Layout}
16108      */
16109     panel : false,
16110     
16111     /**
16112      * @property layout
16113      * for BC  - use el in new code
16114      * @type {Roo.Layout}
16115      */
16116     layout : false,
16117     
16118      /**
16119      * @cfg {Function|boolean} disabled
16120      * If this module is disabled by some rule, return true from the funtion
16121      */
16122     disabled : false,
16123     
16124     /**
16125      * @cfg {String} parent 
16126      * Name of parent element which it get xtype added to..
16127      */
16128     parent: false,
16129     
16130     /**
16131      * @cfg {String} order
16132      * Used to set the order in which elements are created (usefull for multiple tabs)
16133      */
16134     
16135     order : false,
16136     /**
16137      * @cfg {String} name
16138      * String to display while loading.
16139      */
16140     name : false,
16141     /**
16142      * @cfg {String} region
16143      * Region to render component to (defaults to center)
16144      */
16145     region : 'center',
16146     
16147     /**
16148      * @cfg {Array} items
16149      * A single item array - the first element is the root of the tree..
16150      * It's done this way to stay compatible with the Xtype system...
16151      */
16152     items : false,
16153     
16154     /**
16155      * @property _tree
16156      * The method that retuns the tree of parts that make up this compoennt 
16157      * @type {function}
16158      */
16159     _tree  : false,
16160     
16161      /**
16162      * render
16163      * render element to dom or tree
16164      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16165      */
16166     
16167     render : function(el)
16168     {
16169         
16170         el = el || false;
16171         var hp = this.parent ? 1 : 0;
16172         Roo.debug &&  Roo.log(this);
16173         
16174         var tree = this._tree ? this._tree() : this.tree();
16175
16176         
16177         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16178             // if parent is a '#.....' string, then let's use that..
16179             var ename = this.parent.substr(1);
16180             this.parent = false;
16181             Roo.debug && Roo.log(ename);
16182             switch (ename) {
16183                 case 'bootstrap-body':
16184                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16185                         // this is the BorderLayout standard?
16186                        this.parent = { el : true };
16187                        break;
16188                     }
16189                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16190                         // need to insert stuff...
16191                         this.parent =  {
16192                              el : new Roo.bootstrap.layout.Border({
16193                                  el : document.body, 
16194                      
16195                                  center: {
16196                                     titlebar: false,
16197                                     autoScroll:false,
16198                                     closeOnTab: true,
16199                                     tabPosition: 'top',
16200                                       //resizeTabs: true,
16201                                     alwaysShowTabs: true,
16202                                     hideTabs: false
16203                                      //minTabWidth: 140
16204                                  }
16205                              })
16206                         
16207                          };
16208                          break;
16209                     }
16210                          
16211                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16212                         this.parent = { el :  new  Roo.bootstrap.Body() };
16213                         Roo.debug && Roo.log("setting el to doc body");
16214                          
16215                     } else {
16216                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16217                     }
16218                     break;
16219                 case 'bootstrap':
16220                     this.parent = { el : true};
16221                     // fall through
16222                 default:
16223                     el = Roo.get(ename);
16224                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16225                         this.parent = { el : true};
16226                     }
16227                     
16228                     break;
16229             }
16230                 
16231             
16232             if (!el && !this.parent) {
16233                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16234                 return;
16235             }
16236         }
16237         
16238         Roo.debug && Roo.log("EL:");
16239         Roo.debug && Roo.log(el);
16240         Roo.debug && Roo.log("this.parent.el:");
16241         Roo.debug && Roo.log(this.parent.el);
16242         
16243
16244         // altertive root elements ??? - we need a better way to indicate these.
16245         var is_alt = Roo.XComponent.is_alt ||
16246                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16247                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16248                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16249         
16250         
16251         
16252         if (!this.parent && is_alt) {
16253             //el = Roo.get(document.body);
16254             this.parent = { el : true };
16255         }
16256             
16257             
16258         
16259         if (!this.parent) {
16260             
16261             Roo.debug && Roo.log("no parent - creating one");
16262             
16263             el = el ? Roo.get(el) : false;      
16264             
16265             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16266                 
16267                 this.parent =  {
16268                     el : new Roo.bootstrap.layout.Border({
16269                         el: el || document.body,
16270                     
16271                         center: {
16272                             titlebar: false,
16273                             autoScroll:false,
16274                             closeOnTab: true,
16275                             tabPosition: 'top',
16276                              //resizeTabs: true,
16277                             alwaysShowTabs: false,
16278                             hideTabs: true,
16279                             minTabWidth: 140,
16280                             overflow: 'visible'
16281                          }
16282                      })
16283                 };
16284             } else {
16285             
16286                 // it's a top level one..
16287                 this.parent =  {
16288                     el : new Roo.BorderLayout(el || document.body, {
16289                         center: {
16290                             titlebar: false,
16291                             autoScroll:false,
16292                             closeOnTab: true,
16293                             tabPosition: 'top',
16294                              //resizeTabs: true,
16295                             alwaysShowTabs: el && hp? false :  true,
16296                             hideTabs: el || !hp ? true :  false,
16297                             minTabWidth: 140
16298                          }
16299                     })
16300                 };
16301             }
16302         }
16303         
16304         if (!this.parent.el) {
16305                 // probably an old style ctor, which has been disabled.
16306                 return;
16307
16308         }
16309                 // The 'tree' method is  '_tree now' 
16310             
16311         tree.region = tree.region || this.region;
16312         var is_body = false;
16313         if (this.parent.el === true) {
16314             // bootstrap... - body..
16315             if (el) {
16316                 tree.el = el;
16317             }
16318             this.parent.el = Roo.factory(tree);
16319             is_body = true;
16320         }
16321         
16322         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16323         this.fireEvent('built', this);
16324         
16325         this.panel = this.el;
16326         this.layout = this.panel.layout;
16327         this.parentLayout = this.parent.layout  || false;  
16328          
16329     }
16330     
16331 });
16332
16333 Roo.apply(Roo.XComponent, {
16334     /**
16335      * @property  hideProgress
16336      * true to disable the building progress bar.. usefull on single page renders.
16337      * @type Boolean
16338      */
16339     hideProgress : false,
16340     /**
16341      * @property  buildCompleted
16342      * True when the builder has completed building the interface.
16343      * @type Boolean
16344      */
16345     buildCompleted : false,
16346      
16347     /**
16348      * @property  topModule
16349      * the upper most module - uses document.element as it's constructor.
16350      * @type Object
16351      */
16352      
16353     topModule  : false,
16354       
16355     /**
16356      * @property  modules
16357      * array of modules to be created by registration system.
16358      * @type {Array} of Roo.XComponent
16359      */
16360     
16361     modules : [],
16362     /**
16363      * @property  elmodules
16364      * array of modules to be created by which use #ID 
16365      * @type {Array} of Roo.XComponent
16366      */
16367      
16368     elmodules : [],
16369
16370      /**
16371      * @property  is_alt
16372      * Is an alternative Root - normally used by bootstrap or other systems,
16373      *    where the top element in the tree can wrap 'body' 
16374      * @type {boolean}  (default false)
16375      */
16376      
16377     is_alt : false,
16378     /**
16379      * @property  build_from_html
16380      * Build elements from html - used by bootstrap HTML stuff 
16381      *    - this is cleared after build is completed
16382      * @type {boolean}    (default false)
16383      */
16384      
16385     build_from_html : false,
16386     /**
16387      * Register components to be built later.
16388      *
16389      * This solves the following issues
16390      * - Building is not done on page load, but after an authentication process has occured.
16391      * - Interface elements are registered on page load
16392      * - Parent Interface elements may not be loaded before child, so this handles that..
16393      * 
16394      *
16395      * example:
16396      * 
16397      * MyApp.register({
16398           order : '000001',
16399           module : 'Pman.Tab.projectMgr',
16400           region : 'center',
16401           parent : 'Pman.layout',
16402           disabled : false,  // or use a function..
16403         })
16404      
16405      * * @param {Object} details about module
16406      */
16407     register : function(obj) {
16408                 
16409         Roo.XComponent.event.fireEvent('register', obj);
16410         switch(typeof(obj.disabled) ) {
16411                 
16412             case 'undefined':
16413                 break;
16414             
16415             case 'function':
16416                 if ( obj.disabled() ) {
16417                         return;
16418                 }
16419                 break;
16420             
16421             default:
16422                 if (obj.disabled || obj.region == '#disabled') {
16423                         return;
16424                 }
16425                 break;
16426         }
16427                 
16428         this.modules.push(obj);
16429          
16430     },
16431     /**
16432      * convert a string to an object..
16433      * eg. 'AAA.BBB' -> finds AAA.BBB
16434
16435      */
16436     
16437     toObject : function(str)
16438     {
16439         if (!str || typeof(str) == 'object') {
16440             return str;
16441         }
16442         if (str.substring(0,1) == '#') {
16443             return str;
16444         }
16445
16446         var ar = str.split('.');
16447         var rt, o;
16448         rt = ar.shift();
16449             /** eval:var:o */
16450         try {
16451             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16452         } catch (e) {
16453             throw "Module not found : " + str;
16454         }
16455         
16456         if (o === false) {
16457             throw "Module not found : " + str;
16458         }
16459         Roo.each(ar, function(e) {
16460             if (typeof(o[e]) == 'undefined') {
16461                 throw "Module not found : " + str;
16462             }
16463             o = o[e];
16464         });
16465         
16466         return o;
16467         
16468     },
16469     
16470     
16471     /**
16472      * move modules into their correct place in the tree..
16473      * 
16474      */
16475     preBuild : function ()
16476     {
16477         var _t = this;
16478         Roo.each(this.modules , function (obj)
16479         {
16480             Roo.XComponent.event.fireEvent('beforebuild', obj);
16481             
16482             var opar = obj.parent;
16483             try { 
16484                 obj.parent = this.toObject(opar);
16485             } catch(e) {
16486                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16487                 return;
16488             }
16489             
16490             if (!obj.parent) {
16491                 Roo.debug && Roo.log("GOT top level module");
16492                 Roo.debug && Roo.log(obj);
16493                 obj.modules = new Roo.util.MixedCollection(false, 
16494                     function(o) { return o.order + '' }
16495                 );
16496                 this.topModule = obj;
16497                 return;
16498             }
16499                         // parent is a string (usually a dom element name..)
16500             if (typeof(obj.parent) == 'string') {
16501                 this.elmodules.push(obj);
16502                 return;
16503             }
16504             if (obj.parent.constructor != Roo.XComponent) {
16505                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16506             }
16507             if (!obj.parent.modules) {
16508                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16509                     function(o) { return o.order + '' }
16510                 );
16511             }
16512             if (obj.parent.disabled) {
16513                 obj.disabled = true;
16514             }
16515             obj.parent.modules.add(obj);
16516         }, this);
16517     },
16518     
16519      /**
16520      * make a list of modules to build.
16521      * @return {Array} list of modules. 
16522      */ 
16523     
16524     buildOrder : function()
16525     {
16526         var _this = this;
16527         var cmp = function(a,b) {   
16528             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16529         };
16530         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16531             throw "No top level modules to build";
16532         }
16533         
16534         // make a flat list in order of modules to build.
16535         var mods = this.topModule ? [ this.topModule ] : [];
16536                 
16537         
16538         // elmodules (is a list of DOM based modules )
16539         Roo.each(this.elmodules, function(e) {
16540             mods.push(e);
16541             if (!this.topModule &&
16542                 typeof(e.parent) == 'string' &&
16543                 e.parent.substring(0,1) == '#' &&
16544                 Roo.get(e.parent.substr(1))
16545                ) {
16546                 
16547                 _this.topModule = e;
16548             }
16549             
16550         });
16551
16552         
16553         // add modules to their parents..
16554         var addMod = function(m) {
16555             Roo.debug && Roo.log("build Order: add: " + m.name);
16556                 
16557             mods.push(m);
16558             if (m.modules && !m.disabled) {
16559                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16560                 m.modules.keySort('ASC',  cmp );
16561                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16562     
16563                 m.modules.each(addMod);
16564             } else {
16565                 Roo.debug && Roo.log("build Order: no child modules");
16566             }
16567             // not sure if this is used any more..
16568             if (m.finalize) {
16569                 m.finalize.name = m.name + " (clean up) ";
16570                 mods.push(m.finalize);
16571             }
16572             
16573         }
16574         if (this.topModule && this.topModule.modules) { 
16575             this.topModule.modules.keySort('ASC',  cmp );
16576             this.topModule.modules.each(addMod);
16577         } 
16578         return mods;
16579     },
16580     
16581      /**
16582      * Build the registered modules.
16583      * @param {Object} parent element.
16584      * @param {Function} optional method to call after module has been added.
16585      * 
16586      */ 
16587    
16588     build : function(opts) 
16589     {
16590         
16591         if (typeof(opts) != 'undefined') {
16592             Roo.apply(this,opts);
16593         }
16594         
16595         this.preBuild();
16596         var mods = this.buildOrder();
16597       
16598         //this.allmods = mods;
16599         //Roo.debug && Roo.log(mods);
16600         //return;
16601         if (!mods.length) { // should not happen
16602             throw "NO modules!!!";
16603         }
16604         
16605         
16606         var msg = "Building Interface...";
16607         // flash it up as modal - so we store the mask!?
16608         if (!this.hideProgress && Roo.MessageBox) {
16609             Roo.MessageBox.show({ title: 'loading' });
16610             Roo.MessageBox.show({
16611                title: "Please wait...",
16612                msg: msg,
16613                width:450,
16614                progress:true,
16615                buttons : false,
16616                closable:false,
16617                modal: false
16618               
16619             });
16620         }
16621         var total = mods.length;
16622         
16623         var _this = this;
16624         var progressRun = function() {
16625             if (!mods.length) {
16626                 Roo.debug && Roo.log('hide?');
16627                 if (!this.hideProgress && Roo.MessageBox) {
16628                     Roo.MessageBox.hide();
16629                 }
16630                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16631                 
16632                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16633                 
16634                 // THE END...
16635                 return false;   
16636             }
16637             
16638             var m = mods.shift();
16639             
16640             
16641             Roo.debug && Roo.log(m);
16642             // not sure if this is supported any more.. - modules that are are just function
16643             if (typeof(m) == 'function') { 
16644                 m.call(this);
16645                 return progressRun.defer(10, _this);
16646             } 
16647             
16648             
16649             msg = "Building Interface " + (total  - mods.length) + 
16650                     " of " + total + 
16651                     (m.name ? (' - ' + m.name) : '');
16652                         Roo.debug && Roo.log(msg);
16653             if (!_this.hideProgress &&  Roo.MessageBox) { 
16654                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16655             }
16656             
16657          
16658             // is the module disabled?
16659             var disabled = (typeof(m.disabled) == 'function') ?
16660                 m.disabled.call(m.module.disabled) : m.disabled;    
16661             
16662             
16663             if (disabled) {
16664                 return progressRun(); // we do not update the display!
16665             }
16666             
16667             // now build 
16668             
16669                         
16670                         
16671             m.render();
16672             // it's 10 on top level, and 1 on others??? why...
16673             return progressRun.defer(10, _this);
16674              
16675         }
16676         progressRun.defer(1, _this);
16677      
16678         
16679         
16680     },
16681     /**
16682      * Overlay a set of modified strings onto a component
16683      * This is dependant on our builder exporting the strings and 'named strings' elements.
16684      * 
16685      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16686      * @param {Object} associative array of 'named' string and it's new value.
16687      * 
16688      */
16689         overlayStrings : function( component, strings )
16690     {
16691         if (typeof(component['_named_strings']) == 'undefined') {
16692             throw "ERROR: component does not have _named_strings";
16693         }
16694         for ( var k in strings ) {
16695             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16696             if (md !== false) {
16697                 component['_strings'][md] = strings[k];
16698             } else {
16699                 Roo.log('could not find named string: ' + k + ' in');
16700                 Roo.log(component);
16701             }
16702             
16703         }
16704         
16705     },
16706     
16707         
16708         /**
16709          * Event Object.
16710          *
16711          *
16712          */
16713         event: false, 
16714     /**
16715          * wrapper for event.on - aliased later..  
16716          * Typically use to register a event handler for register:
16717          *
16718          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16719          *
16720          */
16721     on : false
16722    
16723     
16724     
16725 });
16726
16727 Roo.XComponent.event = new Roo.util.Observable({
16728                 events : { 
16729                         /**
16730                          * @event register
16731                          * Fires when an Component is registered,
16732                          * set the disable property on the Component to stop registration.
16733                          * @param {Roo.XComponent} c the component being registerd.
16734                          * 
16735                          */
16736                         'register' : true,
16737             /**
16738                          * @event beforebuild
16739                          * Fires before each Component is built
16740                          * can be used to apply permissions.
16741                          * @param {Roo.XComponent} c the component being registerd.
16742                          * 
16743                          */
16744                         'beforebuild' : true,
16745                         /**
16746                          * @event buildcomplete
16747                          * Fires on the top level element when all elements have been built
16748                          * @param {Roo.XComponent} the top level component.
16749                          */
16750                         'buildcomplete' : true
16751                         
16752                 }
16753 });
16754
16755 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16756  //
16757  /**
16758  * marked - a markdown parser
16759  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16760  * https://github.com/chjj/marked
16761  */
16762
16763
16764 /**
16765  *
16766  * Roo.Markdown - is a very crude wrapper around marked..
16767  *
16768  * usage:
16769  * 
16770  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16771  * 
16772  * Note: move the sample code to the bottom of this
16773  * file before uncommenting it.
16774  *
16775  */
16776
16777 Roo.Markdown = {};
16778 Roo.Markdown.toHtml = function(text) {
16779     
16780     var c = new Roo.Markdown.marked.setOptions({
16781             renderer: new Roo.Markdown.marked.Renderer(),
16782             gfm: true,
16783             tables: true,
16784             breaks: false,
16785             pedantic: false,
16786             sanitize: false,
16787             smartLists: true,
16788             smartypants: false
16789           });
16790     // A FEW HACKS!!?
16791     
16792     text = text.replace(/\\\n/g,' ');
16793     return Roo.Markdown.marked(text);
16794 };
16795 //
16796 // converter
16797 //
16798 // Wraps all "globals" so that the only thing
16799 // exposed is makeHtml().
16800 //
16801 (function() {
16802     
16803     /**
16804      * Block-Level Grammar
16805      */
16806     
16807     var block = {
16808       newline: /^\n+/,
16809       code: /^( {4}[^\n]+\n*)+/,
16810       fences: noop,
16811       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16812       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16813       nptable: noop,
16814       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16815       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16816       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16817       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16818       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16819       table: noop,
16820       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16821       text: /^[^\n]+/
16822     };
16823     
16824     block.bullet = /(?:[*+-]|\d+\.)/;
16825     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16826     block.item = replace(block.item, 'gm')
16827       (/bull/g, block.bullet)
16828       ();
16829     
16830     block.list = replace(block.list)
16831       (/bull/g, block.bullet)
16832       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16833       ('def', '\\n+(?=' + block.def.source + ')')
16834       ();
16835     
16836     block.blockquote = replace(block.blockquote)
16837       ('def', block.def)
16838       ();
16839     
16840     block._tag = '(?!(?:'
16841       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16842       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16843       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16844     
16845     block.html = replace(block.html)
16846       ('comment', /<!--[\s\S]*?-->/)
16847       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16848       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16849       (/tag/g, block._tag)
16850       ();
16851     
16852     block.paragraph = replace(block.paragraph)
16853       ('hr', block.hr)
16854       ('heading', block.heading)
16855       ('lheading', block.lheading)
16856       ('blockquote', block.blockquote)
16857       ('tag', '<' + block._tag)
16858       ('def', block.def)
16859       ();
16860     
16861     /**
16862      * Normal Block Grammar
16863      */
16864     
16865     block.normal = merge({}, block);
16866     
16867     /**
16868      * GFM Block Grammar
16869      */
16870     
16871     block.gfm = merge({}, block.normal, {
16872       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16873       paragraph: /^/,
16874       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16875     });
16876     
16877     block.gfm.paragraph = replace(block.paragraph)
16878       ('(?!', '(?!'
16879         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16880         + block.list.source.replace('\\1', '\\3') + '|')
16881       ();
16882     
16883     /**
16884      * GFM + Tables Block Grammar
16885      */
16886     
16887     block.tables = merge({}, block.gfm, {
16888       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16889       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16890     });
16891     
16892     /**
16893      * Block Lexer
16894      */
16895     
16896     function Lexer(options) {
16897       this.tokens = [];
16898       this.tokens.links = {};
16899       this.options = options || marked.defaults;
16900       this.rules = block.normal;
16901     
16902       if (this.options.gfm) {
16903         if (this.options.tables) {
16904           this.rules = block.tables;
16905         } else {
16906           this.rules = block.gfm;
16907         }
16908       }
16909     }
16910     
16911     /**
16912      * Expose Block Rules
16913      */
16914     
16915     Lexer.rules = block;
16916     
16917     /**
16918      * Static Lex Method
16919      */
16920     
16921     Lexer.lex = function(src, options) {
16922       var lexer = new Lexer(options);
16923       return lexer.lex(src);
16924     };
16925     
16926     /**
16927      * Preprocessing
16928      */
16929     
16930     Lexer.prototype.lex = function(src) {
16931       src = src
16932         .replace(/\r\n|\r/g, '\n')
16933         .replace(/\t/g, '    ')
16934         .replace(/\u00a0/g, ' ')
16935         .replace(/\u2424/g, '\n');
16936     
16937       return this.token(src, true);
16938     };
16939     
16940     /**
16941      * Lexing
16942      */
16943     
16944     Lexer.prototype.token = function(src, top, bq) {
16945       var src = src.replace(/^ +$/gm, '')
16946         , next
16947         , loose
16948         , cap
16949         , bull
16950         , b
16951         , item
16952         , space
16953         , i
16954         , l;
16955     
16956       while (src) {
16957         // newline
16958         if (cap = this.rules.newline.exec(src)) {
16959           src = src.substring(cap[0].length);
16960           if (cap[0].length > 1) {
16961             this.tokens.push({
16962               type: 'space'
16963             });
16964           }
16965         }
16966     
16967         // code
16968         if (cap = this.rules.code.exec(src)) {
16969           src = src.substring(cap[0].length);
16970           cap = cap[0].replace(/^ {4}/gm, '');
16971           this.tokens.push({
16972             type: 'code',
16973             text: !this.options.pedantic
16974               ? cap.replace(/\n+$/, '')
16975               : cap
16976           });
16977           continue;
16978         }
16979     
16980         // fences (gfm)
16981         if (cap = this.rules.fences.exec(src)) {
16982           src = src.substring(cap[0].length);
16983           this.tokens.push({
16984             type: 'code',
16985             lang: cap[2],
16986             text: cap[3] || ''
16987           });
16988           continue;
16989         }
16990     
16991         // heading
16992         if (cap = this.rules.heading.exec(src)) {
16993           src = src.substring(cap[0].length);
16994           this.tokens.push({
16995             type: 'heading',
16996             depth: cap[1].length,
16997             text: cap[2]
16998           });
16999           continue;
17000         }
17001     
17002         // table no leading pipe (gfm)
17003         if (top && (cap = this.rules.nptable.exec(src))) {
17004           src = src.substring(cap[0].length);
17005     
17006           item = {
17007             type: 'table',
17008             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17009             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17010             cells: cap[3].replace(/\n$/, '').split('\n')
17011           };
17012     
17013           for (i = 0; i < item.align.length; i++) {
17014             if (/^ *-+: *$/.test(item.align[i])) {
17015               item.align[i] = 'right';
17016             } else if (/^ *:-+: *$/.test(item.align[i])) {
17017               item.align[i] = 'center';
17018             } else if (/^ *:-+ *$/.test(item.align[i])) {
17019               item.align[i] = 'left';
17020             } else {
17021               item.align[i] = null;
17022             }
17023           }
17024     
17025           for (i = 0; i < item.cells.length; i++) {
17026             item.cells[i] = item.cells[i].split(/ *\| */);
17027           }
17028     
17029           this.tokens.push(item);
17030     
17031           continue;
17032         }
17033     
17034         // lheading
17035         if (cap = this.rules.lheading.exec(src)) {
17036           src = src.substring(cap[0].length);
17037           this.tokens.push({
17038             type: 'heading',
17039             depth: cap[2] === '=' ? 1 : 2,
17040             text: cap[1]
17041           });
17042           continue;
17043         }
17044     
17045         // hr
17046         if (cap = this.rules.hr.exec(src)) {
17047           src = src.substring(cap[0].length);
17048           this.tokens.push({
17049             type: 'hr'
17050           });
17051           continue;
17052         }
17053     
17054         // blockquote
17055         if (cap = this.rules.blockquote.exec(src)) {
17056           src = src.substring(cap[0].length);
17057     
17058           this.tokens.push({
17059             type: 'blockquote_start'
17060           });
17061     
17062           cap = cap[0].replace(/^ *> ?/gm, '');
17063     
17064           // Pass `top` to keep the current
17065           // "toplevel" state. This is exactly
17066           // how markdown.pl works.
17067           this.token(cap, top, true);
17068     
17069           this.tokens.push({
17070             type: 'blockquote_end'
17071           });
17072     
17073           continue;
17074         }
17075     
17076         // list
17077         if (cap = this.rules.list.exec(src)) {
17078           src = src.substring(cap[0].length);
17079           bull = cap[2];
17080     
17081           this.tokens.push({
17082             type: 'list_start',
17083             ordered: bull.length > 1
17084           });
17085     
17086           // Get each top-level item.
17087           cap = cap[0].match(this.rules.item);
17088     
17089           next = false;
17090           l = cap.length;
17091           i = 0;
17092     
17093           for (; i < l; i++) {
17094             item = cap[i];
17095     
17096             // Remove the list item's bullet
17097             // so it is seen as the next token.
17098             space = item.length;
17099             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17100     
17101             // Outdent whatever the
17102             // list item contains. Hacky.
17103             if (~item.indexOf('\n ')) {
17104               space -= item.length;
17105               item = !this.options.pedantic
17106                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17107                 : item.replace(/^ {1,4}/gm, '');
17108             }
17109     
17110             // Determine whether the next list item belongs here.
17111             // Backpedal if it does not belong in this list.
17112             if (this.options.smartLists && i !== l - 1) {
17113               b = block.bullet.exec(cap[i + 1])[0];
17114               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17115                 src = cap.slice(i + 1).join('\n') + src;
17116                 i = l - 1;
17117               }
17118             }
17119     
17120             // Determine whether item is loose or not.
17121             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17122             // for discount behavior.
17123             loose = next || /\n\n(?!\s*$)/.test(item);
17124             if (i !== l - 1) {
17125               next = item.charAt(item.length - 1) === '\n';
17126               if (!loose) { loose = next; }
17127             }
17128     
17129             this.tokens.push({
17130               type: loose
17131                 ? 'loose_item_start'
17132                 : 'list_item_start'
17133             });
17134     
17135             // Recurse.
17136             this.token(item, false, bq);
17137     
17138             this.tokens.push({
17139               type: 'list_item_end'
17140             });
17141           }
17142     
17143           this.tokens.push({
17144             type: 'list_end'
17145           });
17146     
17147           continue;
17148         }
17149     
17150         // html
17151         if (cap = this.rules.html.exec(src)) {
17152           src = src.substring(cap[0].length);
17153           this.tokens.push({
17154             type: this.options.sanitize
17155               ? 'paragraph'
17156               : 'html',
17157             pre: !this.options.sanitizer
17158               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17159             text: cap[0]
17160           });
17161           continue;
17162         }
17163     
17164         // def
17165         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17166           src = src.substring(cap[0].length);
17167           this.tokens.links[cap[1].toLowerCase()] = {
17168             href: cap[2],
17169             title: cap[3]
17170           };
17171           continue;
17172         }
17173     
17174         // table (gfm)
17175         if (top && (cap = this.rules.table.exec(src))) {
17176           src = src.substring(cap[0].length);
17177     
17178           item = {
17179             type: 'table',
17180             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17181             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17182             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17183           };
17184     
17185           for (i = 0; i < item.align.length; i++) {
17186             if (/^ *-+: *$/.test(item.align[i])) {
17187               item.align[i] = 'right';
17188             } else if (/^ *:-+: *$/.test(item.align[i])) {
17189               item.align[i] = 'center';
17190             } else if (/^ *:-+ *$/.test(item.align[i])) {
17191               item.align[i] = 'left';
17192             } else {
17193               item.align[i] = null;
17194             }
17195           }
17196     
17197           for (i = 0; i < item.cells.length; i++) {
17198             item.cells[i] = item.cells[i]
17199               .replace(/^ *\| *| *\| *$/g, '')
17200               .split(/ *\| */);
17201           }
17202     
17203           this.tokens.push(item);
17204     
17205           continue;
17206         }
17207     
17208         // top-level paragraph
17209         if (top && (cap = this.rules.paragraph.exec(src))) {
17210           src = src.substring(cap[0].length);
17211           this.tokens.push({
17212             type: 'paragraph',
17213             text: cap[1].charAt(cap[1].length - 1) === '\n'
17214               ? cap[1].slice(0, -1)
17215               : cap[1]
17216           });
17217           continue;
17218         }
17219     
17220         // text
17221         if (cap = this.rules.text.exec(src)) {
17222           // Top-level should never reach here.
17223           src = src.substring(cap[0].length);
17224           this.tokens.push({
17225             type: 'text',
17226             text: cap[0]
17227           });
17228           continue;
17229         }
17230     
17231         if (src) {
17232           throw new
17233             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17234         }
17235       }
17236     
17237       return this.tokens;
17238     };
17239     
17240     /**
17241      * Inline-Level Grammar
17242      */
17243     
17244     var inline = {
17245       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17246       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17247       url: noop,
17248       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17249       link: /^!?\[(inside)\]\(href\)/,
17250       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17251       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17252       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17253       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17254       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17255       br: /^ {2,}\n(?!\s*$)/,
17256       del: noop,
17257       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17258     };
17259     
17260     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17261     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17262     
17263     inline.link = replace(inline.link)
17264       ('inside', inline._inside)
17265       ('href', inline._href)
17266       ();
17267     
17268     inline.reflink = replace(inline.reflink)
17269       ('inside', inline._inside)
17270       ();
17271     
17272     /**
17273      * Normal Inline Grammar
17274      */
17275     
17276     inline.normal = merge({}, inline);
17277     
17278     /**
17279      * Pedantic Inline Grammar
17280      */
17281     
17282     inline.pedantic = merge({}, inline.normal, {
17283       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17284       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17285     });
17286     
17287     /**
17288      * GFM Inline Grammar
17289      */
17290     
17291     inline.gfm = merge({}, inline.normal, {
17292       escape: replace(inline.escape)('])', '~|])')(),
17293       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17294       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17295       text: replace(inline.text)
17296         (']|', '~]|')
17297         ('|', '|https?://|')
17298         ()
17299     });
17300     
17301     /**
17302      * GFM + Line Breaks Inline Grammar
17303      */
17304     
17305     inline.breaks = merge({}, inline.gfm, {
17306       br: replace(inline.br)('{2,}', '*')(),
17307       text: replace(inline.gfm.text)('{2,}', '*')()
17308     });
17309     
17310     /**
17311      * Inline Lexer & Compiler
17312      */
17313     
17314     function InlineLexer(links, options) {
17315       this.options = options || marked.defaults;
17316       this.links = links;
17317       this.rules = inline.normal;
17318       this.renderer = this.options.renderer || new Renderer;
17319       this.renderer.options = this.options;
17320     
17321       if (!this.links) {
17322         throw new
17323           Error('Tokens array requires a `links` property.');
17324       }
17325     
17326       if (this.options.gfm) {
17327         if (this.options.breaks) {
17328           this.rules = inline.breaks;
17329         } else {
17330           this.rules = inline.gfm;
17331         }
17332       } else if (this.options.pedantic) {
17333         this.rules = inline.pedantic;
17334       }
17335     }
17336     
17337     /**
17338      * Expose Inline Rules
17339      */
17340     
17341     InlineLexer.rules = inline;
17342     
17343     /**
17344      * Static Lexing/Compiling Method
17345      */
17346     
17347     InlineLexer.output = function(src, links, options) {
17348       var inline = new InlineLexer(links, options);
17349       return inline.output(src);
17350     };
17351     
17352     /**
17353      * Lexing/Compiling
17354      */
17355     
17356     InlineLexer.prototype.output = function(src) {
17357       var out = ''
17358         , link
17359         , text
17360         , href
17361         , cap;
17362     
17363       while (src) {
17364         // escape
17365         if (cap = this.rules.escape.exec(src)) {
17366           src = src.substring(cap[0].length);
17367           out += cap[1];
17368           continue;
17369         }
17370     
17371         // autolink
17372         if (cap = this.rules.autolink.exec(src)) {
17373           src = src.substring(cap[0].length);
17374           if (cap[2] === '@') {
17375             text = cap[1].charAt(6) === ':'
17376               ? this.mangle(cap[1].substring(7))
17377               : this.mangle(cap[1]);
17378             href = this.mangle('mailto:') + text;
17379           } else {
17380             text = escape(cap[1]);
17381             href = text;
17382           }
17383           out += this.renderer.link(href, null, text);
17384           continue;
17385         }
17386     
17387         // url (gfm)
17388         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17389           src = src.substring(cap[0].length);
17390           text = escape(cap[1]);
17391           href = text;
17392           out += this.renderer.link(href, null, text);
17393           continue;
17394         }
17395     
17396         // tag
17397         if (cap = this.rules.tag.exec(src)) {
17398           if (!this.inLink && /^<a /i.test(cap[0])) {
17399             this.inLink = true;
17400           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17401             this.inLink = false;
17402           }
17403           src = src.substring(cap[0].length);
17404           out += this.options.sanitize
17405             ? this.options.sanitizer
17406               ? this.options.sanitizer(cap[0])
17407               : escape(cap[0])
17408             : cap[0];
17409           continue;
17410         }
17411     
17412         // link
17413         if (cap = this.rules.link.exec(src)) {
17414           src = src.substring(cap[0].length);
17415           this.inLink = true;
17416           out += this.outputLink(cap, {
17417             href: cap[2],
17418             title: cap[3]
17419           });
17420           this.inLink = false;
17421           continue;
17422         }
17423     
17424         // reflink, nolink
17425         if ((cap = this.rules.reflink.exec(src))
17426             || (cap = this.rules.nolink.exec(src))) {
17427           src = src.substring(cap[0].length);
17428           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17429           link = this.links[link.toLowerCase()];
17430           if (!link || !link.href) {
17431             out += cap[0].charAt(0);
17432             src = cap[0].substring(1) + src;
17433             continue;
17434           }
17435           this.inLink = true;
17436           out += this.outputLink(cap, link);
17437           this.inLink = false;
17438           continue;
17439         }
17440     
17441         // strong
17442         if (cap = this.rules.strong.exec(src)) {
17443           src = src.substring(cap[0].length);
17444           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17445           continue;
17446         }
17447     
17448         // em
17449         if (cap = this.rules.em.exec(src)) {
17450           src = src.substring(cap[0].length);
17451           out += this.renderer.em(this.output(cap[2] || cap[1]));
17452           continue;
17453         }
17454     
17455         // code
17456         if (cap = this.rules.code.exec(src)) {
17457           src = src.substring(cap[0].length);
17458           out += this.renderer.codespan(escape(cap[2], true));
17459           continue;
17460         }
17461     
17462         // br
17463         if (cap = this.rules.br.exec(src)) {
17464           src = src.substring(cap[0].length);
17465           out += this.renderer.br();
17466           continue;
17467         }
17468     
17469         // del (gfm)
17470         if (cap = this.rules.del.exec(src)) {
17471           src = src.substring(cap[0].length);
17472           out += this.renderer.del(this.output(cap[1]));
17473           continue;
17474         }
17475     
17476         // text
17477         if (cap = this.rules.text.exec(src)) {
17478           src = src.substring(cap[0].length);
17479           out += this.renderer.text(escape(this.smartypants(cap[0])));
17480           continue;
17481         }
17482     
17483         if (src) {
17484           throw new
17485             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17486         }
17487       }
17488     
17489       return out;
17490     };
17491     
17492     /**
17493      * Compile Link
17494      */
17495     
17496     InlineLexer.prototype.outputLink = function(cap, link) {
17497       var href = escape(link.href)
17498         , title = link.title ? escape(link.title) : null;
17499     
17500       return cap[0].charAt(0) !== '!'
17501         ? this.renderer.link(href, title, this.output(cap[1]))
17502         : this.renderer.image(href, title, escape(cap[1]));
17503     };
17504     
17505     /**
17506      * Smartypants Transformations
17507      */
17508     
17509     InlineLexer.prototype.smartypants = function(text) {
17510       if (!this.options.smartypants)  { return text; }
17511       return text
17512         // em-dashes
17513         .replace(/---/g, '\u2014')
17514         // en-dashes
17515         .replace(/--/g, '\u2013')
17516         // opening singles
17517         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17518         // closing singles & apostrophes
17519         .replace(/'/g, '\u2019')
17520         // opening doubles
17521         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17522         // closing doubles
17523         .replace(/"/g, '\u201d')
17524         // ellipses
17525         .replace(/\.{3}/g, '\u2026');
17526     };
17527     
17528     /**
17529      * Mangle Links
17530      */
17531     
17532     InlineLexer.prototype.mangle = function(text) {
17533       if (!this.options.mangle) { return text; }
17534       var out = ''
17535         , l = text.length
17536         , i = 0
17537         , ch;
17538     
17539       for (; i < l; i++) {
17540         ch = text.charCodeAt(i);
17541         if (Math.random() > 0.5) {
17542           ch = 'x' + ch.toString(16);
17543         }
17544         out += '&#' + ch + ';';
17545       }
17546     
17547       return out;
17548     };
17549     
17550     /**
17551      * Renderer
17552      */
17553     
17554     function Renderer(options) {
17555       this.options = options || {};
17556     }
17557     
17558     Renderer.prototype.code = function(code, lang, escaped) {
17559       if (this.options.highlight) {
17560         var out = this.options.highlight(code, lang);
17561         if (out != null && out !== code) {
17562           escaped = true;
17563           code = out;
17564         }
17565       } else {
17566             // hack!!! - it's already escapeD?
17567             escaped = true;
17568       }
17569     
17570       if (!lang) {
17571         return '<pre><code>'
17572           + (escaped ? code : escape(code, true))
17573           + '\n</code></pre>';
17574       }
17575     
17576       return '<pre><code class="'
17577         + this.options.langPrefix
17578         + escape(lang, true)
17579         + '">'
17580         + (escaped ? code : escape(code, true))
17581         + '\n</code></pre>\n';
17582     };
17583     
17584     Renderer.prototype.blockquote = function(quote) {
17585       return '<blockquote>\n' + quote + '</blockquote>\n';
17586     };
17587     
17588     Renderer.prototype.html = function(html) {
17589       return html;
17590     };
17591     
17592     Renderer.prototype.heading = function(text, level, raw) {
17593       return '<h'
17594         + level
17595         + ' id="'
17596         + this.options.headerPrefix
17597         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17598         + '">'
17599         + text
17600         + '</h'
17601         + level
17602         + '>\n';
17603     };
17604     
17605     Renderer.prototype.hr = function() {
17606       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17607     };
17608     
17609     Renderer.prototype.list = function(body, ordered) {
17610       var type = ordered ? 'ol' : 'ul';
17611       return '<' + type + '>\n' + body + '</' + type + '>\n';
17612     };
17613     
17614     Renderer.prototype.listitem = function(text) {
17615       return '<li>' + text + '</li>\n';
17616     };
17617     
17618     Renderer.prototype.paragraph = function(text) {
17619       return '<p>' + text + '</p>\n';
17620     };
17621     
17622     Renderer.prototype.table = function(header, body) {
17623       return '<table class="table table-striped">\n'
17624         + '<thead>\n'
17625         + header
17626         + '</thead>\n'
17627         + '<tbody>\n'
17628         + body
17629         + '</tbody>\n'
17630         + '</table>\n';
17631     };
17632     
17633     Renderer.prototype.tablerow = function(content) {
17634       return '<tr>\n' + content + '</tr>\n';
17635     };
17636     
17637     Renderer.prototype.tablecell = function(content, flags) {
17638       var type = flags.header ? 'th' : 'td';
17639       var tag = flags.align
17640         ? '<' + type + ' style="text-align:' + flags.align + '">'
17641         : '<' + type + '>';
17642       return tag + content + '</' + type + '>\n';
17643     };
17644     
17645     // span level renderer
17646     Renderer.prototype.strong = function(text) {
17647       return '<strong>' + text + '</strong>';
17648     };
17649     
17650     Renderer.prototype.em = function(text) {
17651       return '<em>' + text + '</em>';
17652     };
17653     
17654     Renderer.prototype.codespan = function(text) {
17655       return '<code>' + text + '</code>';
17656     };
17657     
17658     Renderer.prototype.br = function() {
17659       return this.options.xhtml ? '<br/>' : '<br>';
17660     };
17661     
17662     Renderer.prototype.del = function(text) {
17663       return '<del>' + text + '</del>';
17664     };
17665     
17666     Renderer.prototype.link = function(href, title, text) {
17667       if (this.options.sanitize) {
17668         try {
17669           var prot = decodeURIComponent(unescape(href))
17670             .replace(/[^\w:]/g, '')
17671             .toLowerCase();
17672         } catch (e) {
17673           return '';
17674         }
17675         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17676           return '';
17677         }
17678       }
17679       var out = '<a href="' + href + '"';
17680       if (title) {
17681         out += ' title="' + title + '"';
17682       }
17683       out += '>' + text + '</a>';
17684       return out;
17685     };
17686     
17687     Renderer.prototype.image = function(href, title, text) {
17688       var out = '<img src="' + href + '" alt="' + text + '"';
17689       if (title) {
17690         out += ' title="' + title + '"';
17691       }
17692       out += this.options.xhtml ? '/>' : '>';
17693       return out;
17694     };
17695     
17696     Renderer.prototype.text = function(text) {
17697       return text;
17698     };
17699     
17700     /**
17701      * Parsing & Compiling
17702      */
17703     
17704     function Parser(options) {
17705       this.tokens = [];
17706       this.token = null;
17707       this.options = options || marked.defaults;
17708       this.options.renderer = this.options.renderer || new Renderer;
17709       this.renderer = this.options.renderer;
17710       this.renderer.options = this.options;
17711     }
17712     
17713     /**
17714      * Static Parse Method
17715      */
17716     
17717     Parser.parse = function(src, options, renderer) {
17718       var parser = new Parser(options, renderer);
17719       return parser.parse(src);
17720     };
17721     
17722     /**
17723      * Parse Loop
17724      */
17725     
17726     Parser.prototype.parse = function(src) {
17727       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17728       this.tokens = src.reverse();
17729     
17730       var out = '';
17731       while (this.next()) {
17732         out += this.tok();
17733       }
17734     
17735       return out;
17736     };
17737     
17738     /**
17739      * Next Token
17740      */
17741     
17742     Parser.prototype.next = function() {
17743       return this.token = this.tokens.pop();
17744     };
17745     
17746     /**
17747      * Preview Next Token
17748      */
17749     
17750     Parser.prototype.peek = function() {
17751       return this.tokens[this.tokens.length - 1] || 0;
17752     };
17753     
17754     /**
17755      * Parse Text Tokens
17756      */
17757     
17758     Parser.prototype.parseText = function() {
17759       var body = this.token.text;
17760     
17761       while (this.peek().type === 'text') {
17762         body += '\n' + this.next().text;
17763       }
17764     
17765       return this.inline.output(body);
17766     };
17767     
17768     /**
17769      * Parse Current Token
17770      */
17771     
17772     Parser.prototype.tok = function() {
17773       switch (this.token.type) {
17774         case 'space': {
17775           return '';
17776         }
17777         case 'hr': {
17778           return this.renderer.hr();
17779         }
17780         case 'heading': {
17781           return this.renderer.heading(
17782             this.inline.output(this.token.text),
17783             this.token.depth,
17784             this.token.text);
17785         }
17786         case 'code': {
17787           return this.renderer.code(this.token.text,
17788             this.token.lang,
17789             this.token.escaped);
17790         }
17791         case 'table': {
17792           var header = ''
17793             , body = ''
17794             , i
17795             , row
17796             , cell
17797             , flags
17798             , j;
17799     
17800           // header
17801           cell = '';
17802           for (i = 0; i < this.token.header.length; i++) {
17803             flags = { header: true, align: this.token.align[i] };
17804             cell += this.renderer.tablecell(
17805               this.inline.output(this.token.header[i]),
17806               { header: true, align: this.token.align[i] }
17807             );
17808           }
17809           header += this.renderer.tablerow(cell);
17810     
17811           for (i = 0; i < this.token.cells.length; i++) {
17812             row = this.token.cells[i];
17813     
17814             cell = '';
17815             for (j = 0; j < row.length; j++) {
17816               cell += this.renderer.tablecell(
17817                 this.inline.output(row[j]),
17818                 { header: false, align: this.token.align[j] }
17819               );
17820             }
17821     
17822             body += this.renderer.tablerow(cell);
17823           }
17824           return this.renderer.table(header, body);
17825         }
17826         case 'blockquote_start': {
17827           var body = '';
17828     
17829           while (this.next().type !== 'blockquote_end') {
17830             body += this.tok();
17831           }
17832     
17833           return this.renderer.blockquote(body);
17834         }
17835         case 'list_start': {
17836           var body = ''
17837             , ordered = this.token.ordered;
17838     
17839           while (this.next().type !== 'list_end') {
17840             body += this.tok();
17841           }
17842     
17843           return this.renderer.list(body, ordered);
17844         }
17845         case 'list_item_start': {
17846           var body = '';
17847     
17848           while (this.next().type !== 'list_item_end') {
17849             body += this.token.type === 'text'
17850               ? this.parseText()
17851               : this.tok();
17852           }
17853     
17854           return this.renderer.listitem(body);
17855         }
17856         case 'loose_item_start': {
17857           var body = '';
17858     
17859           while (this.next().type !== 'list_item_end') {
17860             body += this.tok();
17861           }
17862     
17863           return this.renderer.listitem(body);
17864         }
17865         case 'html': {
17866           var html = !this.token.pre && !this.options.pedantic
17867             ? this.inline.output(this.token.text)
17868             : this.token.text;
17869           return this.renderer.html(html);
17870         }
17871         case 'paragraph': {
17872           return this.renderer.paragraph(this.inline.output(this.token.text));
17873         }
17874         case 'text': {
17875           return this.renderer.paragraph(this.parseText());
17876         }
17877       }
17878     };
17879     
17880     /**
17881      * Helpers
17882      */
17883     
17884     function escape(html, encode) {
17885       return html
17886         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
17887         .replace(/</g, '&lt;')
17888         .replace(/>/g, '&gt;')
17889         .replace(/"/g, '&quot;')
17890         .replace(/'/g, '&#39;');
17891     }
17892     
17893     function unescape(html) {
17894         // explicitly match decimal, hex, and named HTML entities 
17895       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
17896         n = n.toLowerCase();
17897         if (n === 'colon') { return ':'; }
17898         if (n.charAt(0) === '#') {
17899           return n.charAt(1) === 'x'
17900             ? String.fromCharCode(parseInt(n.substring(2), 16))
17901             : String.fromCharCode(+n.substring(1));
17902         }
17903         return '';
17904       });
17905     }
17906     
17907     function replace(regex, opt) {
17908       regex = regex.source;
17909       opt = opt || '';
17910       return function self(name, val) {
17911         if (!name) { return new RegExp(regex, opt); }
17912         val = val.source || val;
17913         val = val.replace(/(^|[^\[])\^/g, '$1');
17914         regex = regex.replace(name, val);
17915         return self;
17916       };
17917     }
17918     
17919     function noop() {}
17920     noop.exec = noop;
17921     
17922     function merge(obj) {
17923       var i = 1
17924         , target
17925         , key;
17926     
17927       for (; i < arguments.length; i++) {
17928         target = arguments[i];
17929         for (key in target) {
17930           if (Object.prototype.hasOwnProperty.call(target, key)) {
17931             obj[key] = target[key];
17932           }
17933         }
17934       }
17935     
17936       return obj;
17937     }
17938     
17939     
17940     /**
17941      * Marked
17942      */
17943     
17944     function marked(src, opt, callback) {
17945       if (callback || typeof opt === 'function') {
17946         if (!callback) {
17947           callback = opt;
17948           opt = null;
17949         }
17950     
17951         opt = merge({}, marked.defaults, opt || {});
17952     
17953         var highlight = opt.highlight
17954           , tokens
17955           , pending
17956           , i = 0;
17957     
17958         try {
17959           tokens = Lexer.lex(src, opt)
17960         } catch (e) {
17961           return callback(e);
17962         }
17963     
17964         pending = tokens.length;
17965     
17966         var done = function(err) {
17967           if (err) {
17968             opt.highlight = highlight;
17969             return callback(err);
17970           }
17971     
17972           var out;
17973     
17974           try {
17975             out = Parser.parse(tokens, opt);
17976           } catch (e) {
17977             err = e;
17978           }
17979     
17980           opt.highlight = highlight;
17981     
17982           return err
17983             ? callback(err)
17984             : callback(null, out);
17985         };
17986     
17987         if (!highlight || highlight.length < 3) {
17988           return done();
17989         }
17990     
17991         delete opt.highlight;
17992     
17993         if (!pending) { return done(); }
17994     
17995         for (; i < tokens.length; i++) {
17996           (function(token) {
17997             if (token.type !== 'code') {
17998               return --pending || done();
17999             }
18000             return highlight(token.text, token.lang, function(err, code) {
18001               if (err) { return done(err); }
18002               if (code == null || code === token.text) {
18003                 return --pending || done();
18004               }
18005               token.text = code;
18006               token.escaped = true;
18007               --pending || done();
18008             });
18009           })(tokens[i]);
18010         }
18011     
18012         return;
18013       }
18014       try {
18015         if (opt) { opt = merge({}, marked.defaults, opt); }
18016         return Parser.parse(Lexer.lex(src, opt), opt);
18017       } catch (e) {
18018         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18019         if ((opt || marked.defaults).silent) {
18020           return '<p>An error occured:</p><pre>'
18021             + escape(e.message + '', true)
18022             + '</pre>';
18023         }
18024         throw e;
18025       }
18026     }
18027     
18028     /**
18029      * Options
18030      */
18031     
18032     marked.options =
18033     marked.setOptions = function(opt) {
18034       merge(marked.defaults, opt);
18035       return marked;
18036     };
18037     
18038     marked.defaults = {
18039       gfm: true,
18040       tables: true,
18041       breaks: false,
18042       pedantic: false,
18043       sanitize: false,
18044       sanitizer: null,
18045       mangle: true,
18046       smartLists: false,
18047       silent: false,
18048       highlight: null,
18049       langPrefix: 'lang-',
18050       smartypants: false,
18051       headerPrefix: '',
18052       renderer: new Renderer,
18053       xhtml: false
18054     };
18055     
18056     /**
18057      * Expose
18058      */
18059     
18060     marked.Parser = Parser;
18061     marked.parser = Parser.parse;
18062     
18063     marked.Renderer = Renderer;
18064     
18065     marked.Lexer = Lexer;
18066     marked.lexer = Lexer.lex;
18067     
18068     marked.InlineLexer = InlineLexer;
18069     marked.inlineLexer = InlineLexer.output;
18070     
18071     marked.parse = marked;
18072     
18073     Roo.Markdown.marked = marked;
18074
18075 })();/*
18076  * Based on:
18077  * Ext JS Library 1.1.1
18078  * Copyright(c) 2006-2007, Ext JS, LLC.
18079  *
18080  * Originally Released Under LGPL - original licence link has changed is not relivant.
18081  *
18082  * Fork - LGPL
18083  * <script type="text/javascript">
18084  */
18085
18086
18087
18088 /*
18089  * These classes are derivatives of the similarly named classes in the YUI Library.
18090  * The original license:
18091  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18092  * Code licensed under the BSD License:
18093  * http://developer.yahoo.net/yui/license.txt
18094  */
18095
18096 (function() {
18097
18098 var Event=Roo.EventManager;
18099 var Dom=Roo.lib.Dom;
18100
18101 /**
18102  * @class Roo.dd.DragDrop
18103  * @extends Roo.util.Observable
18104  * Defines the interface and base operation of items that that can be
18105  * dragged or can be drop targets.  It was designed to be extended, overriding
18106  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18107  * Up to three html elements can be associated with a DragDrop instance:
18108  * <ul>
18109  * <li>linked element: the element that is passed into the constructor.
18110  * This is the element which defines the boundaries for interaction with
18111  * other DragDrop objects.</li>
18112  * <li>handle element(s): The drag operation only occurs if the element that
18113  * was clicked matches a handle element.  By default this is the linked
18114  * element, but there are times that you will want only a portion of the
18115  * linked element to initiate the drag operation, and the setHandleElId()
18116  * method provides a way to define this.</li>
18117  * <li>drag element: this represents the element that would be moved along
18118  * with the cursor during a drag operation.  By default, this is the linked
18119  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18120  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18121  * </li>
18122  * </ul>
18123  * This class should not be instantiated until the onload event to ensure that
18124  * the associated elements are available.
18125  * The following would define a DragDrop obj that would interact with any
18126  * other DragDrop obj in the "group1" group:
18127  * <pre>
18128  *  dd = new Roo.dd.DragDrop("div1", "group1");
18129  * </pre>
18130  * Since none of the event handlers have been implemented, nothing would
18131  * actually happen if you were to run the code above.  Normally you would
18132  * override this class or one of the default implementations, but you can
18133  * also override the methods you want on an instance of the class...
18134  * <pre>
18135  *  dd.onDragDrop = function(e, id) {
18136  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18137  *  }
18138  * </pre>
18139  * @constructor
18140  * @param {String} id of the element that is linked to this instance
18141  * @param {String} sGroup the group of related DragDrop objects
18142  * @param {object} config an object containing configurable attributes
18143  *                Valid properties for DragDrop:
18144  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18145  */
18146 Roo.dd.DragDrop = function(id, sGroup, config) {
18147     if (id) {
18148         this.init(id, sGroup, config);
18149     }
18150     
18151 };
18152
18153 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18154
18155     /**
18156      * The id of the element associated with this object.  This is what we
18157      * refer to as the "linked element" because the size and position of
18158      * this element is used to determine when the drag and drop objects have
18159      * interacted.
18160      * @property id
18161      * @type String
18162      */
18163     id: null,
18164
18165     /**
18166      * Configuration attributes passed into the constructor
18167      * @property config
18168      * @type object
18169      */
18170     config: null,
18171
18172     /**
18173      * The id of the element that will be dragged.  By default this is same
18174      * as the linked element , but could be changed to another element. Ex:
18175      * Roo.dd.DDProxy
18176      * @property dragElId
18177      * @type String
18178      * @private
18179      */
18180     dragElId: null,
18181
18182     /**
18183      * the id of the element that initiates the drag operation.  By default
18184      * this is the linked element, but could be changed to be a child of this
18185      * element.  This lets us do things like only starting the drag when the
18186      * header element within the linked html element is clicked.
18187      * @property handleElId
18188      * @type String
18189      * @private
18190      */
18191     handleElId: null,
18192
18193     /**
18194      * An associative array of HTML tags that will be ignored if clicked.
18195      * @property invalidHandleTypes
18196      * @type {string: string}
18197      */
18198     invalidHandleTypes: null,
18199
18200     /**
18201      * An associative array of ids for elements that will be ignored if clicked
18202      * @property invalidHandleIds
18203      * @type {string: string}
18204      */
18205     invalidHandleIds: null,
18206
18207     /**
18208      * An indexted array of css class names for elements that will be ignored
18209      * if clicked.
18210      * @property invalidHandleClasses
18211      * @type string[]
18212      */
18213     invalidHandleClasses: null,
18214
18215     /**
18216      * The linked element's absolute X position at the time the drag was
18217      * started
18218      * @property startPageX
18219      * @type int
18220      * @private
18221      */
18222     startPageX: 0,
18223
18224     /**
18225      * The linked element's absolute X position at the time the drag was
18226      * started
18227      * @property startPageY
18228      * @type int
18229      * @private
18230      */
18231     startPageY: 0,
18232
18233     /**
18234      * The group defines a logical collection of DragDrop objects that are
18235      * related.  Instances only get events when interacting with other
18236      * DragDrop object in the same group.  This lets us define multiple
18237      * groups using a single DragDrop subclass if we want.
18238      * @property groups
18239      * @type {string: string}
18240      */
18241     groups: null,
18242
18243     /**
18244      * Individual drag/drop instances can be locked.  This will prevent
18245      * onmousedown start drag.
18246      * @property locked
18247      * @type boolean
18248      * @private
18249      */
18250     locked: false,
18251
18252     /**
18253      * Lock this instance
18254      * @method lock
18255      */
18256     lock: function() { this.locked = true; },
18257
18258     /**
18259      * Unlock this instace
18260      * @method unlock
18261      */
18262     unlock: function() { this.locked = false; },
18263
18264     /**
18265      * By default, all insances can be a drop target.  This can be disabled by
18266      * setting isTarget to false.
18267      * @method isTarget
18268      * @type boolean
18269      */
18270     isTarget: true,
18271
18272     /**
18273      * The padding configured for this drag and drop object for calculating
18274      * the drop zone intersection with this object.
18275      * @method padding
18276      * @type int[]
18277      */
18278     padding: null,
18279
18280     /**
18281      * Cached reference to the linked element
18282      * @property _domRef
18283      * @private
18284      */
18285     _domRef: null,
18286
18287     /**
18288      * Internal typeof flag
18289      * @property __ygDragDrop
18290      * @private
18291      */
18292     __ygDragDrop: true,
18293
18294     /**
18295      * Set to true when horizontal contraints are applied
18296      * @property constrainX
18297      * @type boolean
18298      * @private
18299      */
18300     constrainX: false,
18301
18302     /**
18303      * Set to true when vertical contraints are applied
18304      * @property constrainY
18305      * @type boolean
18306      * @private
18307      */
18308     constrainY: false,
18309
18310     /**
18311      * The left constraint
18312      * @property minX
18313      * @type int
18314      * @private
18315      */
18316     minX: 0,
18317
18318     /**
18319      * The right constraint
18320      * @property maxX
18321      * @type int
18322      * @private
18323      */
18324     maxX: 0,
18325
18326     /**
18327      * The up constraint
18328      * @property minY
18329      * @type int
18330      * @type int
18331      * @private
18332      */
18333     minY: 0,
18334
18335     /**
18336      * The down constraint
18337      * @property maxY
18338      * @type int
18339      * @private
18340      */
18341     maxY: 0,
18342
18343     /**
18344      * Maintain offsets when we resetconstraints.  Set to true when you want
18345      * the position of the element relative to its parent to stay the same
18346      * when the page changes
18347      *
18348      * @property maintainOffset
18349      * @type boolean
18350      */
18351     maintainOffset: false,
18352
18353     /**
18354      * Array of pixel locations the element will snap to if we specified a
18355      * horizontal graduation/interval.  This array is generated automatically
18356      * when you define a tick interval.
18357      * @property xTicks
18358      * @type int[]
18359      */
18360     xTicks: null,
18361
18362     /**
18363      * Array of pixel locations the element will snap to if we specified a
18364      * vertical graduation/interval.  This array is generated automatically
18365      * when you define a tick interval.
18366      * @property yTicks
18367      * @type int[]
18368      */
18369     yTicks: null,
18370
18371     /**
18372      * By default the drag and drop instance will only respond to the primary
18373      * button click (left button for a right-handed mouse).  Set to true to
18374      * allow drag and drop to start with any mouse click that is propogated
18375      * by the browser
18376      * @property primaryButtonOnly
18377      * @type boolean
18378      */
18379     primaryButtonOnly: true,
18380
18381     /**
18382      * The availabe property is false until the linked dom element is accessible.
18383      * @property available
18384      * @type boolean
18385      */
18386     available: false,
18387
18388     /**
18389      * By default, drags can only be initiated if the mousedown occurs in the
18390      * region the linked element is.  This is done in part to work around a
18391      * bug in some browsers that mis-report the mousedown if the previous
18392      * mouseup happened outside of the window.  This property is set to true
18393      * if outer handles are defined.
18394      *
18395      * @property hasOuterHandles
18396      * @type boolean
18397      * @default false
18398      */
18399     hasOuterHandles: false,
18400
18401     /**
18402      * Code that executes immediately before the startDrag event
18403      * @method b4StartDrag
18404      * @private
18405      */
18406     b4StartDrag: function(x, y) { },
18407
18408     /**
18409      * Abstract method called after a drag/drop object is clicked
18410      * and the drag or mousedown time thresholds have beeen met.
18411      * @method startDrag
18412      * @param {int} X click location
18413      * @param {int} Y click location
18414      */
18415     startDrag: function(x, y) { /* override this */ },
18416
18417     /**
18418      * Code that executes immediately before the onDrag event
18419      * @method b4Drag
18420      * @private
18421      */
18422     b4Drag: function(e) { },
18423
18424     /**
18425      * Abstract method called during the onMouseMove event while dragging an
18426      * object.
18427      * @method onDrag
18428      * @param {Event} e the mousemove event
18429      */
18430     onDrag: function(e) { /* override this */ },
18431
18432     /**
18433      * Abstract method called when this element fist begins hovering over
18434      * another DragDrop obj
18435      * @method onDragEnter
18436      * @param {Event} e the mousemove event
18437      * @param {String|DragDrop[]} id In POINT mode, the element
18438      * id this is hovering over.  In INTERSECT mode, an array of one or more
18439      * dragdrop items being hovered over.
18440      */
18441     onDragEnter: function(e, id) { /* override this */ },
18442
18443     /**
18444      * Code that executes immediately before the onDragOver event
18445      * @method b4DragOver
18446      * @private
18447      */
18448     b4DragOver: function(e) { },
18449
18450     /**
18451      * Abstract method called when this element is hovering over another
18452      * DragDrop obj
18453      * @method onDragOver
18454      * @param {Event} e the mousemove event
18455      * @param {String|DragDrop[]} id In POINT mode, the element
18456      * id this is hovering over.  In INTERSECT mode, an array of dd items
18457      * being hovered over.
18458      */
18459     onDragOver: function(e, id) { /* override this */ },
18460
18461     /**
18462      * Code that executes immediately before the onDragOut event
18463      * @method b4DragOut
18464      * @private
18465      */
18466     b4DragOut: function(e) { },
18467
18468     /**
18469      * Abstract method called when we are no longer hovering over an element
18470      * @method onDragOut
18471      * @param {Event} e the mousemove event
18472      * @param {String|DragDrop[]} id In POINT mode, the element
18473      * id this was hovering over.  In INTERSECT mode, an array of dd items
18474      * that the mouse is no longer over.
18475      */
18476     onDragOut: function(e, id) { /* override this */ },
18477
18478     /**
18479      * Code that executes immediately before the onDragDrop event
18480      * @method b4DragDrop
18481      * @private
18482      */
18483     b4DragDrop: function(e) { },
18484
18485     /**
18486      * Abstract method called when this item is dropped on another DragDrop
18487      * obj
18488      * @method onDragDrop
18489      * @param {Event} e the mouseup event
18490      * @param {String|DragDrop[]} id In POINT mode, the element
18491      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18492      * was dropped on.
18493      */
18494     onDragDrop: function(e, id) { /* override this */ },
18495
18496     /**
18497      * Abstract method called when this item is dropped on an area with no
18498      * drop target
18499      * @method onInvalidDrop
18500      * @param {Event} e the mouseup event
18501      */
18502     onInvalidDrop: function(e) { /* override this */ },
18503
18504     /**
18505      * Code that executes immediately before the endDrag event
18506      * @method b4EndDrag
18507      * @private
18508      */
18509     b4EndDrag: function(e) { },
18510
18511     /**
18512      * Fired when we are done dragging the object
18513      * @method endDrag
18514      * @param {Event} e the mouseup event
18515      */
18516     endDrag: function(e) { /* override this */ },
18517
18518     /**
18519      * Code executed immediately before the onMouseDown event
18520      * @method b4MouseDown
18521      * @param {Event} e the mousedown event
18522      * @private
18523      */
18524     b4MouseDown: function(e) {  },
18525
18526     /**
18527      * Event handler that fires when a drag/drop obj gets a mousedown
18528      * @method onMouseDown
18529      * @param {Event} e the mousedown event
18530      */
18531     onMouseDown: function(e) { /* override this */ },
18532
18533     /**
18534      * Event handler that fires when a drag/drop obj gets a mouseup
18535      * @method onMouseUp
18536      * @param {Event} e the mouseup event
18537      */
18538     onMouseUp: function(e) { /* override this */ },
18539
18540     /**
18541      * Override the onAvailable method to do what is needed after the initial
18542      * position was determined.
18543      * @method onAvailable
18544      */
18545     onAvailable: function () {
18546     },
18547
18548     /*
18549      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18550      * @type Object
18551      */
18552     defaultPadding : {left:0, right:0, top:0, bottom:0},
18553
18554     /*
18555      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18556  *
18557  * Usage:
18558  <pre><code>
18559  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18560                 { dragElId: "existingProxyDiv" });
18561  dd.startDrag = function(){
18562      this.constrainTo("parent-id");
18563  };
18564  </code></pre>
18565  * Or you can initalize it using the {@link Roo.Element} object:
18566  <pre><code>
18567  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18568      startDrag : function(){
18569          this.constrainTo("parent-id");
18570      }
18571  });
18572  </code></pre>
18573      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18574      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18575      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18576      * an object containing the sides to pad. For example: {right:10, bottom:10}
18577      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18578      */
18579     constrainTo : function(constrainTo, pad, inContent){
18580         if(typeof pad == "number"){
18581             pad = {left: pad, right:pad, top:pad, bottom:pad};
18582         }
18583         pad = pad || this.defaultPadding;
18584         var b = Roo.get(this.getEl()).getBox();
18585         var ce = Roo.get(constrainTo);
18586         var s = ce.getScroll();
18587         var c, cd = ce.dom;
18588         if(cd == document.body){
18589             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18590         }else{
18591             xy = ce.getXY();
18592             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18593         }
18594
18595
18596         var topSpace = b.y - c.y;
18597         var leftSpace = b.x - c.x;
18598
18599         this.resetConstraints();
18600         this.setXConstraint(leftSpace - (pad.left||0), // left
18601                 c.width - leftSpace - b.width - (pad.right||0) //right
18602         );
18603         this.setYConstraint(topSpace - (pad.top||0), //top
18604                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18605         );
18606     },
18607
18608     /**
18609      * Returns a reference to the linked element
18610      * @method getEl
18611      * @return {HTMLElement} the html element
18612      */
18613     getEl: function() {
18614         if (!this._domRef) {
18615             this._domRef = Roo.getDom(this.id);
18616         }
18617
18618         return this._domRef;
18619     },
18620
18621     /**
18622      * Returns a reference to the actual element to drag.  By default this is
18623      * the same as the html element, but it can be assigned to another
18624      * element. An example of this can be found in Roo.dd.DDProxy
18625      * @method getDragEl
18626      * @return {HTMLElement} the html element
18627      */
18628     getDragEl: function() {
18629         return Roo.getDom(this.dragElId);
18630     },
18631
18632     /**
18633      * Sets up the DragDrop object.  Must be called in the constructor of any
18634      * Roo.dd.DragDrop subclass
18635      * @method init
18636      * @param id the id of the linked element
18637      * @param {String} sGroup the group of related items
18638      * @param {object} config configuration attributes
18639      */
18640     init: function(id, sGroup, config) {
18641         this.initTarget(id, sGroup, config);
18642         if (!Roo.isTouch) {
18643             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18644         }
18645         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18646         // Event.on(this.id, "selectstart", Event.preventDefault);
18647     },
18648
18649     /**
18650      * Initializes Targeting functionality only... the object does not
18651      * get a mousedown handler.
18652      * @method initTarget
18653      * @param id the id of the linked element
18654      * @param {String} sGroup the group of related items
18655      * @param {object} config configuration attributes
18656      */
18657     initTarget: function(id, sGroup, config) {
18658
18659         // configuration attributes
18660         this.config = config || {};
18661
18662         // create a local reference to the drag and drop manager
18663         this.DDM = Roo.dd.DDM;
18664         // initialize the groups array
18665         this.groups = {};
18666
18667         // assume that we have an element reference instead of an id if the
18668         // parameter is not a string
18669         if (typeof id !== "string") {
18670             id = Roo.id(id);
18671         }
18672
18673         // set the id
18674         this.id = id;
18675
18676         // add to an interaction group
18677         this.addToGroup((sGroup) ? sGroup : "default");
18678
18679         // We don't want to register this as the handle with the manager
18680         // so we just set the id rather than calling the setter.
18681         this.handleElId = id;
18682
18683         // the linked element is the element that gets dragged by default
18684         this.setDragElId(id);
18685
18686         // by default, clicked anchors will not start drag operations.
18687         this.invalidHandleTypes = { A: "A" };
18688         this.invalidHandleIds = {};
18689         this.invalidHandleClasses = [];
18690
18691         this.applyConfig();
18692
18693         this.handleOnAvailable();
18694     },
18695
18696     /**
18697      * Applies the configuration parameters that were passed into the constructor.
18698      * This is supposed to happen at each level through the inheritance chain.  So
18699      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18700      * DragDrop in order to get all of the parameters that are available in
18701      * each object.
18702      * @method applyConfig
18703      */
18704     applyConfig: function() {
18705
18706         // configurable properties:
18707         //    padding, isTarget, maintainOffset, primaryButtonOnly
18708         this.padding           = this.config.padding || [0, 0, 0, 0];
18709         this.isTarget          = (this.config.isTarget !== false);
18710         this.maintainOffset    = (this.config.maintainOffset);
18711         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18712
18713     },
18714
18715     /**
18716      * Executed when the linked element is available
18717      * @method handleOnAvailable
18718      * @private
18719      */
18720     handleOnAvailable: function() {
18721         this.available = true;
18722         this.resetConstraints();
18723         this.onAvailable();
18724     },
18725
18726      /**
18727      * Configures the padding for the target zone in px.  Effectively expands
18728      * (or reduces) the virtual object size for targeting calculations.
18729      * Supports css-style shorthand; if only one parameter is passed, all sides
18730      * will have that padding, and if only two are passed, the top and bottom
18731      * will have the first param, the left and right the second.
18732      * @method setPadding
18733      * @param {int} iTop    Top pad
18734      * @param {int} iRight  Right pad
18735      * @param {int} iBot    Bot pad
18736      * @param {int} iLeft   Left pad
18737      */
18738     setPadding: function(iTop, iRight, iBot, iLeft) {
18739         // this.padding = [iLeft, iRight, iTop, iBot];
18740         if (!iRight && 0 !== iRight) {
18741             this.padding = [iTop, iTop, iTop, iTop];
18742         } else if (!iBot && 0 !== iBot) {
18743             this.padding = [iTop, iRight, iTop, iRight];
18744         } else {
18745             this.padding = [iTop, iRight, iBot, iLeft];
18746         }
18747     },
18748
18749     /**
18750      * Stores the initial placement of the linked element.
18751      * @method setInitialPosition
18752      * @param {int} diffX   the X offset, default 0
18753      * @param {int} diffY   the Y offset, default 0
18754      */
18755     setInitPosition: function(diffX, diffY) {
18756         var el = this.getEl();
18757
18758         if (!this.DDM.verifyEl(el)) {
18759             return;
18760         }
18761
18762         var dx = diffX || 0;
18763         var dy = diffY || 0;
18764
18765         var p = Dom.getXY( el );
18766
18767         this.initPageX = p[0] - dx;
18768         this.initPageY = p[1] - dy;
18769
18770         this.lastPageX = p[0];
18771         this.lastPageY = p[1];
18772
18773
18774         this.setStartPosition(p);
18775     },
18776
18777     /**
18778      * Sets the start position of the element.  This is set when the obj
18779      * is initialized, the reset when a drag is started.
18780      * @method setStartPosition
18781      * @param pos current position (from previous lookup)
18782      * @private
18783      */
18784     setStartPosition: function(pos) {
18785         var p = pos || Dom.getXY( this.getEl() );
18786         this.deltaSetXY = null;
18787
18788         this.startPageX = p[0];
18789         this.startPageY = p[1];
18790     },
18791
18792     /**
18793      * Add this instance to a group of related drag/drop objects.  All
18794      * instances belong to at least one group, and can belong to as many
18795      * groups as needed.
18796      * @method addToGroup
18797      * @param sGroup {string} the name of the group
18798      */
18799     addToGroup: function(sGroup) {
18800         this.groups[sGroup] = true;
18801         this.DDM.regDragDrop(this, sGroup);
18802     },
18803
18804     /**
18805      * Remove's this instance from the supplied interaction group
18806      * @method removeFromGroup
18807      * @param {string}  sGroup  The group to drop
18808      */
18809     removeFromGroup: function(sGroup) {
18810         if (this.groups[sGroup]) {
18811             delete this.groups[sGroup];
18812         }
18813
18814         this.DDM.removeDDFromGroup(this, sGroup);
18815     },
18816
18817     /**
18818      * Allows you to specify that an element other than the linked element
18819      * will be moved with the cursor during a drag
18820      * @method setDragElId
18821      * @param id {string} the id of the element that will be used to initiate the drag
18822      */
18823     setDragElId: function(id) {
18824         this.dragElId = id;
18825     },
18826
18827     /**
18828      * Allows you to specify a child of the linked element that should be
18829      * used to initiate the drag operation.  An example of this would be if
18830      * you have a content div with text and links.  Clicking anywhere in the
18831      * content area would normally start the drag operation.  Use this method
18832      * to specify that an element inside of the content div is the element
18833      * that starts the drag operation.
18834      * @method setHandleElId
18835      * @param id {string} the id of the element that will be used to
18836      * initiate the drag.
18837      */
18838     setHandleElId: function(id) {
18839         if (typeof id !== "string") {
18840             id = Roo.id(id);
18841         }
18842         this.handleElId = id;
18843         this.DDM.regHandle(this.id, id);
18844     },
18845
18846     /**
18847      * Allows you to set an element outside of the linked element as a drag
18848      * handle
18849      * @method setOuterHandleElId
18850      * @param id the id of the element that will be used to initiate the drag
18851      */
18852     setOuterHandleElId: function(id) {
18853         if (typeof id !== "string") {
18854             id = Roo.id(id);
18855         }
18856         Event.on(id, "mousedown",
18857                 this.handleMouseDown, this);
18858         this.setHandleElId(id);
18859
18860         this.hasOuterHandles = true;
18861     },
18862
18863     /**
18864      * Remove all drag and drop hooks for this element
18865      * @method unreg
18866      */
18867     unreg: function() {
18868         Event.un(this.id, "mousedown",
18869                 this.handleMouseDown);
18870         Event.un(this.id, "touchstart",
18871                 this.handleMouseDown);
18872         this._domRef = null;
18873         this.DDM._remove(this);
18874     },
18875
18876     destroy : function(){
18877         this.unreg();
18878     },
18879
18880     /**
18881      * Returns true if this instance is locked, or the drag drop mgr is locked
18882      * (meaning that all drag/drop is disabled on the page.)
18883      * @method isLocked
18884      * @return {boolean} true if this obj or all drag/drop is locked, else
18885      * false
18886      */
18887     isLocked: function() {
18888         return (this.DDM.isLocked() || this.locked);
18889     },
18890
18891     /**
18892      * Fired when this object is clicked
18893      * @method handleMouseDown
18894      * @param {Event} e
18895      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18896      * @private
18897      */
18898     handleMouseDown: function(e, oDD){
18899      
18900         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18901             //Roo.log('not touch/ button !=0');
18902             return;
18903         }
18904         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18905             return; // double touch..
18906         }
18907         
18908
18909         if (this.isLocked()) {
18910             //Roo.log('locked');
18911             return;
18912         }
18913
18914         this.DDM.refreshCache(this.groups);
18915 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18916         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18917         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18918             //Roo.log('no outer handes or not over target');
18919                 // do nothing.
18920         } else {
18921 //            Roo.log('check validator');
18922             if (this.clickValidator(e)) {
18923 //                Roo.log('validate success');
18924                 // set the initial element position
18925                 this.setStartPosition();
18926
18927
18928                 this.b4MouseDown(e);
18929                 this.onMouseDown(e);
18930
18931                 this.DDM.handleMouseDown(e, this);
18932
18933                 this.DDM.stopEvent(e);
18934             } else {
18935
18936
18937             }
18938         }
18939     },
18940
18941     clickValidator: function(e) {
18942         var target = e.getTarget();
18943         return ( this.isValidHandleChild(target) &&
18944                     (this.id == this.handleElId ||
18945                         this.DDM.handleWasClicked(target, this.id)) );
18946     },
18947
18948     /**
18949      * Allows you to specify a tag name that should not start a drag operation
18950      * when clicked.  This is designed to facilitate embedding links within a
18951      * drag handle that do something other than start the drag.
18952      * @method addInvalidHandleType
18953      * @param {string} tagName the type of element to exclude
18954      */
18955     addInvalidHandleType: function(tagName) {
18956         var type = tagName.toUpperCase();
18957         this.invalidHandleTypes[type] = type;
18958     },
18959
18960     /**
18961      * Lets you to specify an element id for a child of a drag handle
18962      * that should not initiate a drag
18963      * @method addInvalidHandleId
18964      * @param {string} id the element id of the element you wish to ignore
18965      */
18966     addInvalidHandleId: function(id) {
18967         if (typeof id !== "string") {
18968             id = Roo.id(id);
18969         }
18970         this.invalidHandleIds[id] = id;
18971     },
18972
18973     /**
18974      * Lets you specify a css class of elements that will not initiate a drag
18975      * @method addInvalidHandleClass
18976      * @param {string} cssClass the class of the elements you wish to ignore
18977      */
18978     addInvalidHandleClass: function(cssClass) {
18979         this.invalidHandleClasses.push(cssClass);
18980     },
18981
18982     /**
18983      * Unsets an excluded tag name set by addInvalidHandleType
18984      * @method removeInvalidHandleType
18985      * @param {string} tagName the type of element to unexclude
18986      */
18987     removeInvalidHandleType: function(tagName) {
18988         var type = tagName.toUpperCase();
18989         // this.invalidHandleTypes[type] = null;
18990         delete this.invalidHandleTypes[type];
18991     },
18992
18993     /**
18994      * Unsets an invalid handle id
18995      * @method removeInvalidHandleId
18996      * @param {string} id the id of the element to re-enable
18997      */
18998     removeInvalidHandleId: function(id) {
18999         if (typeof id !== "string") {
19000             id = Roo.id(id);
19001         }
19002         delete this.invalidHandleIds[id];
19003     },
19004
19005     /**
19006      * Unsets an invalid css class
19007      * @method removeInvalidHandleClass
19008      * @param {string} cssClass the class of the element(s) you wish to
19009      * re-enable
19010      */
19011     removeInvalidHandleClass: function(cssClass) {
19012         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19013             if (this.invalidHandleClasses[i] == cssClass) {
19014                 delete this.invalidHandleClasses[i];
19015             }
19016         }
19017     },
19018
19019     /**
19020      * Checks the tag exclusion list to see if this click should be ignored
19021      * @method isValidHandleChild
19022      * @param {HTMLElement} node the HTMLElement to evaluate
19023      * @return {boolean} true if this is a valid tag type, false if not
19024      */
19025     isValidHandleChild: function(node) {
19026
19027         var valid = true;
19028         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19029         var nodeName;
19030         try {
19031             nodeName = node.nodeName.toUpperCase();
19032         } catch(e) {
19033             nodeName = node.nodeName;
19034         }
19035         valid = valid && !this.invalidHandleTypes[nodeName];
19036         valid = valid && !this.invalidHandleIds[node.id];
19037
19038         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19039             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19040         }
19041
19042
19043         return valid;
19044
19045     },
19046
19047     /**
19048      * Create the array of horizontal tick marks if an interval was specified
19049      * in setXConstraint().
19050      * @method setXTicks
19051      * @private
19052      */
19053     setXTicks: function(iStartX, iTickSize) {
19054         this.xTicks = [];
19055         this.xTickSize = iTickSize;
19056
19057         var tickMap = {};
19058
19059         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19060             if (!tickMap[i]) {
19061                 this.xTicks[this.xTicks.length] = i;
19062                 tickMap[i] = true;
19063             }
19064         }
19065
19066         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19067             if (!tickMap[i]) {
19068                 this.xTicks[this.xTicks.length] = i;
19069                 tickMap[i] = true;
19070             }
19071         }
19072
19073         this.xTicks.sort(this.DDM.numericSort) ;
19074     },
19075
19076     /**
19077      * Create the array of vertical tick marks if an interval was specified in
19078      * setYConstraint().
19079      * @method setYTicks
19080      * @private
19081      */
19082     setYTicks: function(iStartY, iTickSize) {
19083         this.yTicks = [];
19084         this.yTickSize = iTickSize;
19085
19086         var tickMap = {};
19087
19088         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19089             if (!tickMap[i]) {
19090                 this.yTicks[this.yTicks.length] = i;
19091                 tickMap[i] = true;
19092             }
19093         }
19094
19095         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19096             if (!tickMap[i]) {
19097                 this.yTicks[this.yTicks.length] = i;
19098                 tickMap[i] = true;
19099             }
19100         }
19101
19102         this.yTicks.sort(this.DDM.numericSort) ;
19103     },
19104
19105     /**
19106      * By default, the element can be dragged any place on the screen.  Use
19107      * this method to limit the horizontal travel of the element.  Pass in
19108      * 0,0 for the parameters if you want to lock the drag to the y axis.
19109      * @method setXConstraint
19110      * @param {int} iLeft the number of pixels the element can move to the left
19111      * @param {int} iRight the number of pixels the element can move to the
19112      * right
19113      * @param {int} iTickSize optional parameter for specifying that the
19114      * element
19115      * should move iTickSize pixels at a time.
19116      */
19117     setXConstraint: function(iLeft, iRight, iTickSize) {
19118         this.leftConstraint = iLeft;
19119         this.rightConstraint = iRight;
19120
19121         this.minX = this.initPageX - iLeft;
19122         this.maxX = this.initPageX + iRight;
19123         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19124
19125         this.constrainX = true;
19126     },
19127
19128     /**
19129      * Clears any constraints applied to this instance.  Also clears ticks
19130      * since they can't exist independent of a constraint at this time.
19131      * @method clearConstraints
19132      */
19133     clearConstraints: function() {
19134         this.constrainX = false;
19135         this.constrainY = false;
19136         this.clearTicks();
19137     },
19138
19139     /**
19140      * Clears any tick interval defined for this instance
19141      * @method clearTicks
19142      */
19143     clearTicks: function() {
19144         this.xTicks = null;
19145         this.yTicks = null;
19146         this.xTickSize = 0;
19147         this.yTickSize = 0;
19148     },
19149
19150     /**
19151      * By default, the element can be dragged any place on the screen.  Set
19152      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19153      * parameters if you want to lock the drag to the x axis.
19154      * @method setYConstraint
19155      * @param {int} iUp the number of pixels the element can move up
19156      * @param {int} iDown the number of pixels the element can move down
19157      * @param {int} iTickSize optional parameter for specifying that the
19158      * element should move iTickSize pixels at a time.
19159      */
19160     setYConstraint: function(iUp, iDown, iTickSize) {
19161         this.topConstraint = iUp;
19162         this.bottomConstraint = iDown;
19163
19164         this.minY = this.initPageY - iUp;
19165         this.maxY = this.initPageY + iDown;
19166         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19167
19168         this.constrainY = true;
19169
19170     },
19171
19172     /**
19173      * resetConstraints must be called if you manually reposition a dd element.
19174      * @method resetConstraints
19175      * @param {boolean} maintainOffset
19176      */
19177     resetConstraints: function() {
19178
19179
19180         // Maintain offsets if necessary
19181         if (this.initPageX || this.initPageX === 0) {
19182             // figure out how much this thing has moved
19183             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19184             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19185
19186             this.setInitPosition(dx, dy);
19187
19188         // This is the first time we have detected the element's position
19189         } else {
19190             this.setInitPosition();
19191         }
19192
19193         if (this.constrainX) {
19194             this.setXConstraint( this.leftConstraint,
19195                                  this.rightConstraint,
19196                                  this.xTickSize        );
19197         }
19198
19199         if (this.constrainY) {
19200             this.setYConstraint( this.topConstraint,
19201                                  this.bottomConstraint,
19202                                  this.yTickSize         );
19203         }
19204     },
19205
19206     /**
19207      * Normally the drag element is moved pixel by pixel, but we can specify
19208      * that it move a number of pixels at a time.  This method resolves the
19209      * location when we have it set up like this.
19210      * @method getTick
19211      * @param {int} val where we want to place the object
19212      * @param {int[]} tickArray sorted array of valid points
19213      * @return {int} the closest tick
19214      * @private
19215      */
19216     getTick: function(val, tickArray) {
19217
19218         if (!tickArray) {
19219             // If tick interval is not defined, it is effectively 1 pixel,
19220             // so we return the value passed to us.
19221             return val;
19222         } else if (tickArray[0] >= val) {
19223             // The value is lower than the first tick, so we return the first
19224             // tick.
19225             return tickArray[0];
19226         } else {
19227             for (var i=0, len=tickArray.length; i<len; ++i) {
19228                 var next = i + 1;
19229                 if (tickArray[next] && tickArray[next] >= val) {
19230                     var diff1 = val - tickArray[i];
19231                     var diff2 = tickArray[next] - val;
19232                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19233                 }
19234             }
19235
19236             // The value is larger than the last tick, so we return the last
19237             // tick.
19238             return tickArray[tickArray.length - 1];
19239         }
19240     },
19241
19242     /**
19243      * toString method
19244      * @method toString
19245      * @return {string} string representation of the dd obj
19246      */
19247     toString: function() {
19248         return ("DragDrop " + this.id);
19249     }
19250
19251 });
19252
19253 })();
19254 /*
19255  * Based on:
19256  * Ext JS Library 1.1.1
19257  * Copyright(c) 2006-2007, Ext JS, LLC.
19258  *
19259  * Originally Released Under LGPL - original licence link has changed is not relivant.
19260  *
19261  * Fork - LGPL
19262  * <script type="text/javascript">
19263  */
19264
19265
19266 /**
19267  * The drag and drop utility provides a framework for building drag and drop
19268  * applications.  In addition to enabling drag and drop for specific elements,
19269  * the drag and drop elements are tracked by the manager class, and the
19270  * interactions between the various elements are tracked during the drag and
19271  * the implementing code is notified about these important moments.
19272  */
19273
19274 // Only load the library once.  Rewriting the manager class would orphan
19275 // existing drag and drop instances.
19276 if (!Roo.dd.DragDropMgr) {
19277
19278 /**
19279  * @class Roo.dd.DragDropMgr
19280  * DragDropMgr is a singleton that tracks the element interaction for
19281  * all DragDrop items in the window.  Generally, you will not call
19282  * this class directly, but it does have helper methods that could
19283  * be useful in your DragDrop implementations.
19284  * @singleton
19285  */
19286 Roo.dd.DragDropMgr = function() {
19287
19288     var Event = Roo.EventManager;
19289
19290     return {
19291
19292         /**
19293          * Two dimensional Array of registered DragDrop objects.  The first
19294          * dimension is the DragDrop item group, the second the DragDrop
19295          * object.
19296          * @property ids
19297          * @type {string: string}
19298          * @private
19299          * @static
19300          */
19301         ids: {},
19302
19303         /**
19304          * Array of element ids defined as drag handles.  Used to determine
19305          * if the element that generated the mousedown event is actually the
19306          * handle and not the html element itself.
19307          * @property handleIds
19308          * @type {string: string}
19309          * @private
19310          * @static
19311          */
19312         handleIds: {},
19313
19314         /**
19315          * the DragDrop object that is currently being dragged
19316          * @property dragCurrent
19317          * @type DragDrop
19318          * @private
19319          * @static
19320          **/
19321         dragCurrent: null,
19322
19323         /**
19324          * the DragDrop object(s) that are being hovered over
19325          * @property dragOvers
19326          * @type Array
19327          * @private
19328          * @static
19329          */
19330         dragOvers: {},
19331
19332         /**
19333          * the X distance between the cursor and the object being dragged
19334          * @property deltaX
19335          * @type int
19336          * @private
19337          * @static
19338          */
19339         deltaX: 0,
19340
19341         /**
19342          * the Y distance between the cursor and the object being dragged
19343          * @property deltaY
19344          * @type int
19345          * @private
19346          * @static
19347          */
19348         deltaY: 0,
19349
19350         /**
19351          * Flag to determine if we should prevent the default behavior of the
19352          * events we define. By default this is true, but this can be set to
19353          * false if you need the default behavior (not recommended)
19354          * @property preventDefault
19355          * @type boolean
19356          * @static
19357          */
19358         preventDefault: true,
19359
19360         /**
19361          * Flag to determine if we should stop the propagation of the events
19362          * we generate. This is true by default but you may want to set it to
19363          * false if the html element contains other features that require the
19364          * mouse click.
19365          * @property stopPropagation
19366          * @type boolean
19367          * @static
19368          */
19369         stopPropagation: true,
19370
19371         /**
19372          * Internal flag that is set to true when drag and drop has been
19373          * intialized
19374          * @property initialized
19375          * @private
19376          * @static
19377          */
19378         initalized: false,
19379
19380         /**
19381          * All drag and drop can be disabled.
19382          * @property locked
19383          * @private
19384          * @static
19385          */
19386         locked: false,
19387
19388         /**
19389          * Called the first time an element is registered.
19390          * @method init
19391          * @private
19392          * @static
19393          */
19394         init: function() {
19395             this.initialized = true;
19396         },
19397
19398         /**
19399          * In point mode, drag and drop interaction is defined by the
19400          * location of the cursor during the drag/drop
19401          * @property POINT
19402          * @type int
19403          * @static
19404          */
19405         POINT: 0,
19406
19407         /**
19408          * In intersect mode, drag and drop interactio nis defined by the
19409          * overlap of two or more drag and drop objects.
19410          * @property INTERSECT
19411          * @type int
19412          * @static
19413          */
19414         INTERSECT: 1,
19415
19416         /**
19417          * The current drag and drop mode.  Default: POINT
19418          * @property mode
19419          * @type int
19420          * @static
19421          */
19422         mode: 0,
19423
19424         /**
19425          * Runs method on all drag and drop objects
19426          * @method _execOnAll
19427          * @private
19428          * @static
19429          */
19430         _execOnAll: function(sMethod, args) {
19431             for (var i in this.ids) {
19432                 for (var j in this.ids[i]) {
19433                     var oDD = this.ids[i][j];
19434                     if (! this.isTypeOfDD(oDD)) {
19435                         continue;
19436                     }
19437                     oDD[sMethod].apply(oDD, args);
19438                 }
19439             }
19440         },
19441
19442         /**
19443          * Drag and drop initialization.  Sets up the global event handlers
19444          * @method _onLoad
19445          * @private
19446          * @static
19447          */
19448         _onLoad: function() {
19449
19450             this.init();
19451
19452             if (!Roo.isTouch) {
19453                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19454                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19455             }
19456             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19457             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19458             
19459             Event.on(window,   "unload",    this._onUnload, this, true);
19460             Event.on(window,   "resize",    this._onResize, this, true);
19461             // Event.on(window,   "mouseout",    this._test);
19462
19463         },
19464
19465         /**
19466          * Reset constraints on all drag and drop objs
19467          * @method _onResize
19468          * @private
19469          * @static
19470          */
19471         _onResize: function(e) {
19472             this._execOnAll("resetConstraints", []);
19473         },
19474
19475         /**
19476          * Lock all drag and drop functionality
19477          * @method lock
19478          * @static
19479          */
19480         lock: function() { this.locked = true; },
19481
19482         /**
19483          * Unlock all drag and drop functionality
19484          * @method unlock
19485          * @static
19486          */
19487         unlock: function() { this.locked = false; },
19488
19489         /**
19490          * Is drag and drop locked?
19491          * @method isLocked
19492          * @return {boolean} True if drag and drop is locked, false otherwise.
19493          * @static
19494          */
19495         isLocked: function() { return this.locked; },
19496
19497         /**
19498          * Location cache that is set for all drag drop objects when a drag is
19499          * initiated, cleared when the drag is finished.
19500          * @property locationCache
19501          * @private
19502          * @static
19503          */
19504         locationCache: {},
19505
19506         /**
19507          * Set useCache to false if you want to force object the lookup of each
19508          * drag and drop linked element constantly during a drag.
19509          * @property useCache
19510          * @type boolean
19511          * @static
19512          */
19513         useCache: true,
19514
19515         /**
19516          * The number of pixels that the mouse needs to move after the
19517          * mousedown before the drag is initiated.  Default=3;
19518          * @property clickPixelThresh
19519          * @type int
19520          * @static
19521          */
19522         clickPixelThresh: 3,
19523
19524         /**
19525          * The number of milliseconds after the mousedown event to initiate the
19526          * drag if we don't get a mouseup event. Default=1000
19527          * @property clickTimeThresh
19528          * @type int
19529          * @static
19530          */
19531         clickTimeThresh: 350,
19532
19533         /**
19534          * Flag that indicates that either the drag pixel threshold or the
19535          * mousdown time threshold has been met
19536          * @property dragThreshMet
19537          * @type boolean
19538          * @private
19539          * @static
19540          */
19541         dragThreshMet: false,
19542
19543         /**
19544          * Timeout used for the click time threshold
19545          * @property clickTimeout
19546          * @type Object
19547          * @private
19548          * @static
19549          */
19550         clickTimeout: null,
19551
19552         /**
19553          * The X position of the mousedown event stored for later use when a
19554          * drag threshold is met.
19555          * @property startX
19556          * @type int
19557          * @private
19558          * @static
19559          */
19560         startX: 0,
19561
19562         /**
19563          * The Y position of the mousedown event stored for later use when a
19564          * drag threshold is met.
19565          * @property startY
19566          * @type int
19567          * @private
19568          * @static
19569          */
19570         startY: 0,
19571
19572         /**
19573          * Each DragDrop instance must be registered with the DragDropMgr.
19574          * This is executed in DragDrop.init()
19575          * @method regDragDrop
19576          * @param {DragDrop} oDD the DragDrop object to register
19577          * @param {String} sGroup the name of the group this element belongs to
19578          * @static
19579          */
19580         regDragDrop: function(oDD, sGroup) {
19581             if (!this.initialized) { this.init(); }
19582
19583             if (!this.ids[sGroup]) {
19584                 this.ids[sGroup] = {};
19585             }
19586             this.ids[sGroup][oDD.id] = oDD;
19587         },
19588
19589         /**
19590          * Removes the supplied dd instance from the supplied group. Executed
19591          * by DragDrop.removeFromGroup, so don't call this function directly.
19592          * @method removeDDFromGroup
19593          * @private
19594          * @static
19595          */
19596         removeDDFromGroup: function(oDD, sGroup) {
19597             if (!this.ids[sGroup]) {
19598                 this.ids[sGroup] = {};
19599             }
19600
19601             var obj = this.ids[sGroup];
19602             if (obj && obj[oDD.id]) {
19603                 delete obj[oDD.id];
19604             }
19605         },
19606
19607         /**
19608          * Unregisters a drag and drop item.  This is executed in
19609          * DragDrop.unreg, use that method instead of calling this directly.
19610          * @method _remove
19611          * @private
19612          * @static
19613          */
19614         _remove: function(oDD) {
19615             for (var g in oDD.groups) {
19616                 if (g && this.ids[g][oDD.id]) {
19617                     delete this.ids[g][oDD.id];
19618                 }
19619             }
19620             delete this.handleIds[oDD.id];
19621         },
19622
19623         /**
19624          * Each DragDrop handle element must be registered.  This is done
19625          * automatically when executing DragDrop.setHandleElId()
19626          * @method regHandle
19627          * @param {String} sDDId the DragDrop id this element is a handle for
19628          * @param {String} sHandleId the id of the element that is the drag
19629          * handle
19630          * @static
19631          */
19632         regHandle: function(sDDId, sHandleId) {
19633             if (!this.handleIds[sDDId]) {
19634                 this.handleIds[sDDId] = {};
19635             }
19636             this.handleIds[sDDId][sHandleId] = sHandleId;
19637         },
19638
19639         /**
19640          * Utility function to determine if a given element has been
19641          * registered as a drag drop item.
19642          * @method isDragDrop
19643          * @param {String} id the element id to check
19644          * @return {boolean} true if this element is a DragDrop item,
19645          * false otherwise
19646          * @static
19647          */
19648         isDragDrop: function(id) {
19649             return ( this.getDDById(id) ) ? true : false;
19650         },
19651
19652         /**
19653          * Returns the drag and drop instances that are in all groups the
19654          * passed in instance belongs to.
19655          * @method getRelated
19656          * @param {DragDrop} p_oDD the obj to get related data for
19657          * @param {boolean} bTargetsOnly if true, only return targetable objs
19658          * @return {DragDrop[]} the related instances
19659          * @static
19660          */
19661         getRelated: function(p_oDD, bTargetsOnly) {
19662             var oDDs = [];
19663             for (var i in p_oDD.groups) {
19664                 for (j in this.ids[i]) {
19665                     var dd = this.ids[i][j];
19666                     if (! this.isTypeOfDD(dd)) {
19667                         continue;
19668                     }
19669                     if (!bTargetsOnly || dd.isTarget) {
19670                         oDDs[oDDs.length] = dd;
19671                     }
19672                 }
19673             }
19674
19675             return oDDs;
19676         },
19677
19678         /**
19679          * Returns true if the specified dd target is a legal target for
19680          * the specifice drag obj
19681          * @method isLegalTarget
19682          * @param {DragDrop} the drag obj
19683          * @param {DragDrop} the target
19684          * @return {boolean} true if the target is a legal target for the
19685          * dd obj
19686          * @static
19687          */
19688         isLegalTarget: function (oDD, oTargetDD) {
19689             var targets = this.getRelated(oDD, true);
19690             for (var i=0, len=targets.length;i<len;++i) {
19691                 if (targets[i].id == oTargetDD.id) {
19692                     return true;
19693                 }
19694             }
19695
19696             return false;
19697         },
19698
19699         /**
19700          * My goal is to be able to transparently determine if an object is
19701          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19702          * returns "object", oDD.constructor.toString() always returns
19703          * "DragDrop" and not the name of the subclass.  So for now it just
19704          * evaluates a well-known variable in DragDrop.
19705          * @method isTypeOfDD
19706          * @param {Object} the object to evaluate
19707          * @return {boolean} true if typeof oDD = DragDrop
19708          * @static
19709          */
19710         isTypeOfDD: function (oDD) {
19711             return (oDD && oDD.__ygDragDrop);
19712         },
19713
19714         /**
19715          * Utility function to determine if a given element has been
19716          * registered as a drag drop handle for the given Drag Drop object.
19717          * @method isHandle
19718          * @param {String} id the element id to check
19719          * @return {boolean} true if this element is a DragDrop handle, false
19720          * otherwise
19721          * @static
19722          */
19723         isHandle: function(sDDId, sHandleId) {
19724             return ( this.handleIds[sDDId] &&
19725                             this.handleIds[sDDId][sHandleId] );
19726         },
19727
19728         /**
19729          * Returns the DragDrop instance for a given id
19730          * @method getDDById
19731          * @param {String} id the id of the DragDrop object
19732          * @return {DragDrop} the drag drop object, null if it is not found
19733          * @static
19734          */
19735         getDDById: function(id) {
19736             for (var i in this.ids) {
19737                 if (this.ids[i][id]) {
19738                     return this.ids[i][id];
19739                 }
19740             }
19741             return null;
19742         },
19743
19744         /**
19745          * Fired after a registered DragDrop object gets the mousedown event.
19746          * Sets up the events required to track the object being dragged
19747          * @method handleMouseDown
19748          * @param {Event} e the event
19749          * @param oDD the DragDrop object being dragged
19750          * @private
19751          * @static
19752          */
19753         handleMouseDown: function(e, oDD) {
19754             if(Roo.QuickTips){
19755                 Roo.QuickTips.disable();
19756             }
19757             this.currentTarget = e.getTarget();
19758
19759             this.dragCurrent = oDD;
19760
19761             var el = oDD.getEl();
19762
19763             // track start position
19764             this.startX = e.getPageX();
19765             this.startY = e.getPageY();
19766
19767             this.deltaX = this.startX - el.offsetLeft;
19768             this.deltaY = this.startY - el.offsetTop;
19769
19770             this.dragThreshMet = false;
19771
19772             this.clickTimeout = setTimeout(
19773                     function() {
19774                         var DDM = Roo.dd.DDM;
19775                         DDM.startDrag(DDM.startX, DDM.startY);
19776                     },
19777                     this.clickTimeThresh );
19778         },
19779
19780         /**
19781          * Fired when either the drag pixel threshol or the mousedown hold
19782          * time threshold has been met.
19783          * @method startDrag
19784          * @param x {int} the X position of the original mousedown
19785          * @param y {int} the Y position of the original mousedown
19786          * @static
19787          */
19788         startDrag: function(x, y) {
19789             clearTimeout(this.clickTimeout);
19790             if (this.dragCurrent) {
19791                 this.dragCurrent.b4StartDrag(x, y);
19792                 this.dragCurrent.startDrag(x, y);
19793             }
19794             this.dragThreshMet = true;
19795         },
19796
19797         /**
19798          * Internal function to handle the mouseup event.  Will be invoked
19799          * from the context of the document.
19800          * @method handleMouseUp
19801          * @param {Event} e the event
19802          * @private
19803          * @static
19804          */
19805         handleMouseUp: function(e) {
19806
19807             if(Roo.QuickTips){
19808                 Roo.QuickTips.enable();
19809             }
19810             if (! this.dragCurrent) {
19811                 return;
19812             }
19813
19814             clearTimeout(this.clickTimeout);
19815
19816             if (this.dragThreshMet) {
19817                 this.fireEvents(e, true);
19818             } else {
19819             }
19820
19821             this.stopDrag(e);
19822
19823             this.stopEvent(e);
19824         },
19825
19826         /**
19827          * Utility to stop event propagation and event default, if these
19828          * features are turned on.
19829          * @method stopEvent
19830          * @param {Event} e the event as returned by this.getEvent()
19831          * @static
19832          */
19833         stopEvent: function(e){
19834             if(this.stopPropagation) {
19835                 e.stopPropagation();
19836             }
19837
19838             if (this.preventDefault) {
19839                 e.preventDefault();
19840             }
19841         },
19842
19843         /**
19844          * Internal function to clean up event handlers after the drag
19845          * operation is complete
19846          * @method stopDrag
19847          * @param {Event} e the event
19848          * @private
19849          * @static
19850          */
19851         stopDrag: function(e) {
19852             // Fire the drag end event for the item that was dragged
19853             if (this.dragCurrent) {
19854                 if (this.dragThreshMet) {
19855                     this.dragCurrent.b4EndDrag(e);
19856                     this.dragCurrent.endDrag(e);
19857                 }
19858
19859                 this.dragCurrent.onMouseUp(e);
19860             }
19861
19862             this.dragCurrent = null;
19863             this.dragOvers = {};
19864         },
19865
19866         /**
19867          * Internal function to handle the mousemove event.  Will be invoked
19868          * from the context of the html element.
19869          *
19870          * @TODO figure out what we can do about mouse events lost when the
19871          * user drags objects beyond the window boundary.  Currently we can
19872          * detect this in internet explorer by verifying that the mouse is
19873          * down during the mousemove event.  Firefox doesn't give us the
19874          * button state on the mousemove event.
19875          * @method handleMouseMove
19876          * @param {Event} e the event
19877          * @private
19878          * @static
19879          */
19880         handleMouseMove: function(e) {
19881             if (! this.dragCurrent) {
19882                 return true;
19883             }
19884
19885             // var button = e.which || e.button;
19886
19887             // check for IE mouseup outside of page boundary
19888             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19889                 this.stopEvent(e);
19890                 return this.handleMouseUp(e);
19891             }
19892
19893             if (!this.dragThreshMet) {
19894                 var diffX = Math.abs(this.startX - e.getPageX());
19895                 var diffY = Math.abs(this.startY - e.getPageY());
19896                 if (diffX > this.clickPixelThresh ||
19897                             diffY > this.clickPixelThresh) {
19898                     this.startDrag(this.startX, this.startY);
19899                 }
19900             }
19901
19902             if (this.dragThreshMet) {
19903                 this.dragCurrent.b4Drag(e);
19904                 this.dragCurrent.onDrag(e);
19905                 if(!this.dragCurrent.moveOnly){
19906                     this.fireEvents(e, false);
19907                 }
19908             }
19909
19910             this.stopEvent(e);
19911
19912             return true;
19913         },
19914
19915         /**
19916          * Iterates over all of the DragDrop elements to find ones we are
19917          * hovering over or dropping on
19918          * @method fireEvents
19919          * @param {Event} e the event
19920          * @param {boolean} isDrop is this a drop op or a mouseover op?
19921          * @private
19922          * @static
19923          */
19924         fireEvents: function(e, isDrop) {
19925             var dc = this.dragCurrent;
19926
19927             // If the user did the mouse up outside of the window, we could
19928             // get here even though we have ended the drag.
19929             if (!dc || dc.isLocked()) {
19930                 return;
19931             }
19932
19933             var pt = e.getPoint();
19934
19935             // cache the previous dragOver array
19936             var oldOvers = [];
19937
19938             var outEvts   = [];
19939             var overEvts  = [];
19940             var dropEvts  = [];
19941             var enterEvts = [];
19942
19943             // Check to see if the object(s) we were hovering over is no longer
19944             // being hovered over so we can fire the onDragOut event
19945             for (var i in this.dragOvers) {
19946
19947                 var ddo = this.dragOvers[i];
19948
19949                 if (! this.isTypeOfDD(ddo)) {
19950                     continue;
19951                 }
19952
19953                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19954                     outEvts.push( ddo );
19955                 }
19956
19957                 oldOvers[i] = true;
19958                 delete this.dragOvers[i];
19959             }
19960
19961             for (var sGroup in dc.groups) {
19962
19963                 if ("string" != typeof sGroup) {
19964                     continue;
19965                 }
19966
19967                 for (i in this.ids[sGroup]) {
19968                     var oDD = this.ids[sGroup][i];
19969                     if (! this.isTypeOfDD(oDD)) {
19970                         continue;
19971                     }
19972
19973                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
19974                         if (this.isOverTarget(pt, oDD, this.mode)) {
19975                             // look for drop interactions
19976                             if (isDrop) {
19977                                 dropEvts.push( oDD );
19978                             // look for drag enter and drag over interactions
19979                             } else {
19980
19981                                 // initial drag over: dragEnter fires
19982                                 if (!oldOvers[oDD.id]) {
19983                                     enterEvts.push( oDD );
19984                                 // subsequent drag overs: dragOver fires
19985                                 } else {
19986                                     overEvts.push( oDD );
19987                                 }
19988
19989                                 this.dragOvers[oDD.id] = oDD;
19990                             }
19991                         }
19992                     }
19993                 }
19994             }
19995
19996             if (this.mode) {
19997                 if (outEvts.length) {
19998                     dc.b4DragOut(e, outEvts);
19999                     dc.onDragOut(e, outEvts);
20000                 }
20001
20002                 if (enterEvts.length) {
20003                     dc.onDragEnter(e, enterEvts);
20004                 }
20005
20006                 if (overEvts.length) {
20007                     dc.b4DragOver(e, overEvts);
20008                     dc.onDragOver(e, overEvts);
20009                 }
20010
20011                 if (dropEvts.length) {
20012                     dc.b4DragDrop(e, dropEvts);
20013                     dc.onDragDrop(e, dropEvts);
20014                 }
20015
20016             } else {
20017                 // fire dragout events
20018                 var len = 0;
20019                 for (i=0, len=outEvts.length; i<len; ++i) {
20020                     dc.b4DragOut(e, outEvts[i].id);
20021                     dc.onDragOut(e, outEvts[i].id);
20022                 }
20023
20024                 // fire enter events
20025                 for (i=0,len=enterEvts.length; i<len; ++i) {
20026                     // dc.b4DragEnter(e, oDD.id);
20027                     dc.onDragEnter(e, enterEvts[i].id);
20028                 }
20029
20030                 // fire over events
20031                 for (i=0,len=overEvts.length; i<len; ++i) {
20032                     dc.b4DragOver(e, overEvts[i].id);
20033                     dc.onDragOver(e, overEvts[i].id);
20034                 }
20035
20036                 // fire drop events
20037                 for (i=0, len=dropEvts.length; i<len; ++i) {
20038                     dc.b4DragDrop(e, dropEvts[i].id);
20039                     dc.onDragDrop(e, dropEvts[i].id);
20040                 }
20041
20042             }
20043
20044             // notify about a drop that did not find a target
20045             if (isDrop && !dropEvts.length) {
20046                 dc.onInvalidDrop(e);
20047             }
20048
20049         },
20050
20051         /**
20052          * Helper function for getting the best match from the list of drag
20053          * and drop objects returned by the drag and drop events when we are
20054          * in INTERSECT mode.  It returns either the first object that the
20055          * cursor is over, or the object that has the greatest overlap with
20056          * the dragged element.
20057          * @method getBestMatch
20058          * @param  {DragDrop[]} dds The array of drag and drop objects
20059          * targeted
20060          * @return {DragDrop}       The best single match
20061          * @static
20062          */
20063         getBestMatch: function(dds) {
20064             var winner = null;
20065             // Return null if the input is not what we expect
20066             //if (!dds || !dds.length || dds.length == 0) {
20067                // winner = null;
20068             // If there is only one item, it wins
20069             //} else if (dds.length == 1) {
20070
20071             var len = dds.length;
20072
20073             if (len == 1) {
20074                 winner = dds[0];
20075             } else {
20076                 // Loop through the targeted items
20077                 for (var i=0; i<len; ++i) {
20078                     var dd = dds[i];
20079                     // If the cursor is over the object, it wins.  If the
20080                     // cursor is over multiple matches, the first one we come
20081                     // to wins.
20082                     if (dd.cursorIsOver) {
20083                         winner = dd;
20084                         break;
20085                     // Otherwise the object with the most overlap wins
20086                     } else {
20087                         if (!winner ||
20088                             winner.overlap.getArea() < dd.overlap.getArea()) {
20089                             winner = dd;
20090                         }
20091                     }
20092                 }
20093             }
20094
20095             return winner;
20096         },
20097
20098         /**
20099          * Refreshes the cache of the top-left and bottom-right points of the
20100          * drag and drop objects in the specified group(s).  This is in the
20101          * format that is stored in the drag and drop instance, so typical
20102          * usage is:
20103          * <code>
20104          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20105          * </code>
20106          * Alternatively:
20107          * <code>
20108          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20109          * </code>
20110          * @TODO this really should be an indexed array.  Alternatively this
20111          * method could accept both.
20112          * @method refreshCache
20113          * @param {Object} groups an associative array of groups to refresh
20114          * @static
20115          */
20116         refreshCache: function(groups) {
20117             for (var sGroup in groups) {
20118                 if ("string" != typeof sGroup) {
20119                     continue;
20120                 }
20121                 for (var i in this.ids[sGroup]) {
20122                     var oDD = this.ids[sGroup][i];
20123
20124                     if (this.isTypeOfDD(oDD)) {
20125                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20126                         var loc = this.getLocation(oDD);
20127                         if (loc) {
20128                             this.locationCache[oDD.id] = loc;
20129                         } else {
20130                             delete this.locationCache[oDD.id];
20131                             // this will unregister the drag and drop object if
20132                             // the element is not in a usable state
20133                             // oDD.unreg();
20134                         }
20135                     }
20136                 }
20137             }
20138         },
20139
20140         /**
20141          * This checks to make sure an element exists and is in the DOM.  The
20142          * main purpose is to handle cases where innerHTML is used to remove
20143          * drag and drop objects from the DOM.  IE provides an 'unspecified
20144          * error' when trying to access the offsetParent of such an element
20145          * @method verifyEl
20146          * @param {HTMLElement} el the element to check
20147          * @return {boolean} true if the element looks usable
20148          * @static
20149          */
20150         verifyEl: function(el) {
20151             if (el) {
20152                 var parent;
20153                 if(Roo.isIE){
20154                     try{
20155                         parent = el.offsetParent;
20156                     }catch(e){}
20157                 }else{
20158                     parent = el.offsetParent;
20159                 }
20160                 if (parent) {
20161                     return true;
20162                 }
20163             }
20164
20165             return false;
20166         },
20167
20168         /**
20169          * Returns a Region object containing the drag and drop element's position
20170          * and size, including the padding configured for it
20171          * @method getLocation
20172          * @param {DragDrop} oDD the drag and drop object to get the
20173          *                       location for
20174          * @return {Roo.lib.Region} a Region object representing the total area
20175          *                             the element occupies, including any padding
20176          *                             the instance is configured for.
20177          * @static
20178          */
20179         getLocation: function(oDD) {
20180             if (! this.isTypeOfDD(oDD)) {
20181                 return null;
20182             }
20183
20184             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20185
20186             try {
20187                 pos= Roo.lib.Dom.getXY(el);
20188             } catch (e) { }
20189
20190             if (!pos) {
20191                 return null;
20192             }
20193
20194             x1 = pos[0];
20195             x2 = x1 + el.offsetWidth;
20196             y1 = pos[1];
20197             y2 = y1 + el.offsetHeight;
20198
20199             t = y1 - oDD.padding[0];
20200             r = x2 + oDD.padding[1];
20201             b = y2 + oDD.padding[2];
20202             l = x1 - oDD.padding[3];
20203
20204             return new Roo.lib.Region( t, r, b, l );
20205         },
20206
20207         /**
20208          * Checks the cursor location to see if it over the target
20209          * @method isOverTarget
20210          * @param {Roo.lib.Point} pt The point to evaluate
20211          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20212          * @return {boolean} true if the mouse is over the target
20213          * @private
20214          * @static
20215          */
20216         isOverTarget: function(pt, oTarget, intersect) {
20217             // use cache if available
20218             var loc = this.locationCache[oTarget.id];
20219             if (!loc || !this.useCache) {
20220                 loc = this.getLocation(oTarget);
20221                 this.locationCache[oTarget.id] = loc;
20222
20223             }
20224
20225             if (!loc) {
20226                 return false;
20227             }
20228
20229             oTarget.cursorIsOver = loc.contains( pt );
20230
20231             // DragDrop is using this as a sanity check for the initial mousedown
20232             // in this case we are done.  In POINT mode, if the drag obj has no
20233             // contraints, we are also done. Otherwise we need to evaluate the
20234             // location of the target as related to the actual location of the
20235             // dragged element.
20236             var dc = this.dragCurrent;
20237             if (!dc || !dc.getTargetCoord ||
20238                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20239                 return oTarget.cursorIsOver;
20240             }
20241
20242             oTarget.overlap = null;
20243
20244             // Get the current location of the drag element, this is the
20245             // location of the mouse event less the delta that represents
20246             // where the original mousedown happened on the element.  We
20247             // need to consider constraints and ticks as well.
20248             var pos = dc.getTargetCoord(pt.x, pt.y);
20249
20250             var el = dc.getDragEl();
20251             var curRegion = new Roo.lib.Region( pos.y,
20252                                                    pos.x + el.offsetWidth,
20253                                                    pos.y + el.offsetHeight,
20254                                                    pos.x );
20255
20256             var overlap = curRegion.intersect(loc);
20257
20258             if (overlap) {
20259                 oTarget.overlap = overlap;
20260                 return (intersect) ? true : oTarget.cursorIsOver;
20261             } else {
20262                 return false;
20263             }
20264         },
20265
20266         /**
20267          * unload event handler
20268          * @method _onUnload
20269          * @private
20270          * @static
20271          */
20272         _onUnload: function(e, me) {
20273             Roo.dd.DragDropMgr.unregAll();
20274         },
20275
20276         /**
20277          * Cleans up the drag and drop events and objects.
20278          * @method unregAll
20279          * @private
20280          * @static
20281          */
20282         unregAll: function() {
20283
20284             if (this.dragCurrent) {
20285                 this.stopDrag();
20286                 this.dragCurrent = null;
20287             }
20288
20289             this._execOnAll("unreg", []);
20290
20291             for (i in this.elementCache) {
20292                 delete this.elementCache[i];
20293             }
20294
20295             this.elementCache = {};
20296             this.ids = {};
20297         },
20298
20299         /**
20300          * A cache of DOM elements
20301          * @property elementCache
20302          * @private
20303          * @static
20304          */
20305         elementCache: {},
20306
20307         /**
20308          * Get the wrapper for the DOM element specified
20309          * @method getElWrapper
20310          * @param {String} id the id of the element to get
20311          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20312          * @private
20313          * @deprecated This wrapper isn't that useful
20314          * @static
20315          */
20316         getElWrapper: function(id) {
20317             var oWrapper = this.elementCache[id];
20318             if (!oWrapper || !oWrapper.el) {
20319                 oWrapper = this.elementCache[id] =
20320                     new this.ElementWrapper(Roo.getDom(id));
20321             }
20322             return oWrapper;
20323         },
20324
20325         /**
20326          * Returns the actual DOM element
20327          * @method getElement
20328          * @param {String} id the id of the elment to get
20329          * @return {Object} The element
20330          * @deprecated use Roo.getDom instead
20331          * @static
20332          */
20333         getElement: function(id) {
20334             return Roo.getDom(id);
20335         },
20336
20337         /**
20338          * Returns the style property for the DOM element (i.e.,
20339          * document.getElById(id).style)
20340          * @method getCss
20341          * @param {String} id the id of the elment to get
20342          * @return {Object} The style property of the element
20343          * @deprecated use Roo.getDom instead
20344          * @static
20345          */
20346         getCss: function(id) {
20347             var el = Roo.getDom(id);
20348             return (el) ? el.style : null;
20349         },
20350
20351         /**
20352          * Inner class for cached elements
20353          * @class DragDropMgr.ElementWrapper
20354          * @for DragDropMgr
20355          * @private
20356          * @deprecated
20357          */
20358         ElementWrapper: function(el) {
20359                 /**
20360                  * The element
20361                  * @property el
20362                  */
20363                 this.el = el || null;
20364                 /**
20365                  * The element id
20366                  * @property id
20367                  */
20368                 this.id = this.el && el.id;
20369                 /**
20370                  * A reference to the style property
20371                  * @property css
20372                  */
20373                 this.css = this.el && el.style;
20374             },
20375
20376         /**
20377          * Returns the X position of an html element
20378          * @method getPosX
20379          * @param el the element for which to get the position
20380          * @return {int} the X coordinate
20381          * @for DragDropMgr
20382          * @deprecated use Roo.lib.Dom.getX instead
20383          * @static
20384          */
20385         getPosX: function(el) {
20386             return Roo.lib.Dom.getX(el);
20387         },
20388
20389         /**
20390          * Returns the Y position of an html element
20391          * @method getPosY
20392          * @param el the element for which to get the position
20393          * @return {int} the Y coordinate
20394          * @deprecated use Roo.lib.Dom.getY instead
20395          * @static
20396          */
20397         getPosY: function(el) {
20398             return Roo.lib.Dom.getY(el);
20399         },
20400
20401         /**
20402          * Swap two nodes.  In IE, we use the native method, for others we
20403          * emulate the IE behavior
20404          * @method swapNode
20405          * @param n1 the first node to swap
20406          * @param n2 the other node to swap
20407          * @static
20408          */
20409         swapNode: function(n1, n2) {
20410             if (n1.swapNode) {
20411                 n1.swapNode(n2);
20412             } else {
20413                 var p = n2.parentNode;
20414                 var s = n2.nextSibling;
20415
20416                 if (s == n1) {
20417                     p.insertBefore(n1, n2);
20418                 } else if (n2 == n1.nextSibling) {
20419                     p.insertBefore(n2, n1);
20420                 } else {
20421                     n1.parentNode.replaceChild(n2, n1);
20422                     p.insertBefore(n1, s);
20423                 }
20424             }
20425         },
20426
20427         /**
20428          * Returns the current scroll position
20429          * @method getScroll
20430          * @private
20431          * @static
20432          */
20433         getScroll: function () {
20434             var t, l, dde=document.documentElement, db=document.body;
20435             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20436                 t = dde.scrollTop;
20437                 l = dde.scrollLeft;
20438             } else if (db) {
20439                 t = db.scrollTop;
20440                 l = db.scrollLeft;
20441             } else {
20442
20443             }
20444             return { top: t, left: l };
20445         },
20446
20447         /**
20448          * Returns the specified element style property
20449          * @method getStyle
20450          * @param {HTMLElement} el          the element
20451          * @param {string}      styleProp   the style property
20452          * @return {string} The value of the style property
20453          * @deprecated use Roo.lib.Dom.getStyle
20454          * @static
20455          */
20456         getStyle: function(el, styleProp) {
20457             return Roo.fly(el).getStyle(styleProp);
20458         },
20459
20460         /**
20461          * Gets the scrollTop
20462          * @method getScrollTop
20463          * @return {int} the document's scrollTop
20464          * @static
20465          */
20466         getScrollTop: function () { return this.getScroll().top; },
20467
20468         /**
20469          * Gets the scrollLeft
20470          * @method getScrollLeft
20471          * @return {int} the document's scrollTop
20472          * @static
20473          */
20474         getScrollLeft: function () { return this.getScroll().left; },
20475
20476         /**
20477          * Sets the x/y position of an element to the location of the
20478          * target element.
20479          * @method moveToEl
20480          * @param {HTMLElement} moveEl      The element to move
20481          * @param {HTMLElement} targetEl    The position reference element
20482          * @static
20483          */
20484         moveToEl: function (moveEl, targetEl) {
20485             var aCoord = Roo.lib.Dom.getXY(targetEl);
20486             Roo.lib.Dom.setXY(moveEl, aCoord);
20487         },
20488
20489         /**
20490          * Numeric array sort function
20491          * @method numericSort
20492          * @static
20493          */
20494         numericSort: function(a, b) { return (a - b); },
20495
20496         /**
20497          * Internal counter
20498          * @property _timeoutCount
20499          * @private
20500          * @static
20501          */
20502         _timeoutCount: 0,
20503
20504         /**
20505          * Trying to make the load order less important.  Without this we get
20506          * an error if this file is loaded before the Event Utility.
20507          * @method _addListeners
20508          * @private
20509          * @static
20510          */
20511         _addListeners: function() {
20512             var DDM = Roo.dd.DDM;
20513             if ( Roo.lib.Event && document ) {
20514                 DDM._onLoad();
20515             } else {
20516                 if (DDM._timeoutCount > 2000) {
20517                 } else {
20518                     setTimeout(DDM._addListeners, 10);
20519                     if (document && document.body) {
20520                         DDM._timeoutCount += 1;
20521                     }
20522                 }
20523             }
20524         },
20525
20526         /**
20527          * Recursively searches the immediate parent and all child nodes for
20528          * the handle element in order to determine wheter or not it was
20529          * clicked.
20530          * @method handleWasClicked
20531          * @param node the html element to inspect
20532          * @static
20533          */
20534         handleWasClicked: function(node, id) {
20535             if (this.isHandle(id, node.id)) {
20536                 return true;
20537             } else {
20538                 // check to see if this is a text node child of the one we want
20539                 var p = node.parentNode;
20540
20541                 while (p) {
20542                     if (this.isHandle(id, p.id)) {
20543                         return true;
20544                     } else {
20545                         p = p.parentNode;
20546                     }
20547                 }
20548             }
20549
20550             return false;
20551         }
20552
20553     };
20554
20555 }();
20556
20557 // shorter alias, save a few bytes
20558 Roo.dd.DDM = Roo.dd.DragDropMgr;
20559 Roo.dd.DDM._addListeners();
20560
20561 }/*
20562  * Based on:
20563  * Ext JS Library 1.1.1
20564  * Copyright(c) 2006-2007, Ext JS, LLC.
20565  *
20566  * Originally Released Under LGPL - original licence link has changed is not relivant.
20567  *
20568  * Fork - LGPL
20569  * <script type="text/javascript">
20570  */
20571
20572 /**
20573  * @class Roo.dd.DD
20574  * A DragDrop implementation where the linked element follows the
20575  * mouse cursor during a drag.
20576  * @extends Roo.dd.DragDrop
20577  * @constructor
20578  * @param {String} id the id of the linked element
20579  * @param {String} sGroup the group of related DragDrop items
20580  * @param {object} config an object containing configurable attributes
20581  *                Valid properties for DD:
20582  *                    scroll
20583  */
20584 Roo.dd.DD = function(id, sGroup, config) {
20585     if (id) {
20586         this.init(id, sGroup, config);
20587     }
20588 };
20589
20590 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20591
20592     /**
20593      * When set to true, the utility automatically tries to scroll the browser
20594      * window wehn a drag and drop element is dragged near the viewport boundary.
20595      * Defaults to true.
20596      * @property scroll
20597      * @type boolean
20598      */
20599     scroll: true,
20600
20601     /**
20602      * Sets the pointer offset to the distance between the linked element's top
20603      * left corner and the location the element was clicked
20604      * @method autoOffset
20605      * @param {int} iPageX the X coordinate of the click
20606      * @param {int} iPageY the Y coordinate of the click
20607      */
20608     autoOffset: function(iPageX, iPageY) {
20609         var x = iPageX - this.startPageX;
20610         var y = iPageY - this.startPageY;
20611         this.setDelta(x, y);
20612     },
20613
20614     /**
20615      * Sets the pointer offset.  You can call this directly to force the
20616      * offset to be in a particular location (e.g., pass in 0,0 to set it
20617      * to the center of the object)
20618      * @method setDelta
20619      * @param {int} iDeltaX the distance from the left
20620      * @param {int} iDeltaY the distance from the top
20621      */
20622     setDelta: function(iDeltaX, iDeltaY) {
20623         this.deltaX = iDeltaX;
20624         this.deltaY = iDeltaY;
20625     },
20626
20627     /**
20628      * Sets the drag element to the location of the mousedown or click event,
20629      * maintaining the cursor location relative to the location on the element
20630      * that was clicked.  Override this if you want to place the element in a
20631      * location other than where the cursor is.
20632      * @method setDragElPos
20633      * @param {int} iPageX the X coordinate of the mousedown or drag event
20634      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20635      */
20636     setDragElPos: function(iPageX, iPageY) {
20637         // the first time we do this, we are going to check to make sure
20638         // the element has css positioning
20639
20640         var el = this.getDragEl();
20641         this.alignElWithMouse(el, iPageX, iPageY);
20642     },
20643
20644     /**
20645      * Sets the element to the location of the mousedown or click event,
20646      * maintaining the cursor location relative to the location on the element
20647      * that was clicked.  Override this if you want to place the element in a
20648      * location other than where the cursor is.
20649      * @method alignElWithMouse
20650      * @param {HTMLElement} el the element to move
20651      * @param {int} iPageX the X coordinate of the mousedown or drag event
20652      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20653      */
20654     alignElWithMouse: function(el, iPageX, iPageY) {
20655         var oCoord = this.getTargetCoord(iPageX, iPageY);
20656         var fly = el.dom ? el : Roo.fly(el);
20657         if (!this.deltaSetXY) {
20658             var aCoord = [oCoord.x, oCoord.y];
20659             fly.setXY(aCoord);
20660             var newLeft = fly.getLeft(true);
20661             var newTop  = fly.getTop(true);
20662             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20663         } else {
20664             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20665         }
20666
20667         this.cachePosition(oCoord.x, oCoord.y);
20668         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20669         return oCoord;
20670     },
20671
20672     /**
20673      * Saves the most recent position so that we can reset the constraints and
20674      * tick marks on-demand.  We need to know this so that we can calculate the
20675      * number of pixels the element is offset from its original position.
20676      * @method cachePosition
20677      * @param iPageX the current x position (optional, this just makes it so we
20678      * don't have to look it up again)
20679      * @param iPageY the current y position (optional, this just makes it so we
20680      * don't have to look it up again)
20681      */
20682     cachePosition: function(iPageX, iPageY) {
20683         if (iPageX) {
20684             this.lastPageX = iPageX;
20685             this.lastPageY = iPageY;
20686         } else {
20687             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20688             this.lastPageX = aCoord[0];
20689             this.lastPageY = aCoord[1];
20690         }
20691     },
20692
20693     /**
20694      * Auto-scroll the window if the dragged object has been moved beyond the
20695      * visible window boundary.
20696      * @method autoScroll
20697      * @param {int} x the drag element's x position
20698      * @param {int} y the drag element's y position
20699      * @param {int} h the height of the drag element
20700      * @param {int} w the width of the drag element
20701      * @private
20702      */
20703     autoScroll: function(x, y, h, w) {
20704
20705         if (this.scroll) {
20706             // The client height
20707             var clientH = Roo.lib.Dom.getViewWidth();
20708
20709             // The client width
20710             var clientW = Roo.lib.Dom.getViewHeight();
20711
20712             // The amt scrolled down
20713             var st = this.DDM.getScrollTop();
20714
20715             // The amt scrolled right
20716             var sl = this.DDM.getScrollLeft();
20717
20718             // Location of the bottom of the element
20719             var bot = h + y;
20720
20721             // Location of the right of the element
20722             var right = w + x;
20723
20724             // The distance from the cursor to the bottom of the visible area,
20725             // adjusted so that we don't scroll if the cursor is beyond the
20726             // element drag constraints
20727             var toBot = (clientH + st - y - this.deltaY);
20728
20729             // The distance from the cursor to the right of the visible area
20730             var toRight = (clientW + sl - x - this.deltaX);
20731
20732
20733             // How close to the edge the cursor must be before we scroll
20734             // var thresh = (document.all) ? 100 : 40;
20735             var thresh = 40;
20736
20737             // How many pixels to scroll per autoscroll op.  This helps to reduce
20738             // clunky scrolling. IE is more sensitive about this ... it needs this
20739             // value to be higher.
20740             var scrAmt = (document.all) ? 80 : 30;
20741
20742             // Scroll down if we are near the bottom of the visible page and the
20743             // obj extends below the crease
20744             if ( bot > clientH && toBot < thresh ) {
20745                 window.scrollTo(sl, st + scrAmt);
20746             }
20747
20748             // Scroll up if the window is scrolled down and the top of the object
20749             // goes above the top border
20750             if ( y < st && st > 0 && y - st < thresh ) {
20751                 window.scrollTo(sl, st - scrAmt);
20752             }
20753
20754             // Scroll right if the obj is beyond the right border and the cursor is
20755             // near the border.
20756             if ( right > clientW && toRight < thresh ) {
20757                 window.scrollTo(sl + scrAmt, st);
20758             }
20759
20760             // Scroll left if the window has been scrolled to the right and the obj
20761             // extends past the left border
20762             if ( x < sl && sl > 0 && x - sl < thresh ) {
20763                 window.scrollTo(sl - scrAmt, st);
20764             }
20765         }
20766     },
20767
20768     /**
20769      * Finds the location the element should be placed if we want to move
20770      * it to where the mouse location less the click offset would place us.
20771      * @method getTargetCoord
20772      * @param {int} iPageX the X coordinate of the click
20773      * @param {int} iPageY the Y coordinate of the click
20774      * @return an object that contains the coordinates (Object.x and Object.y)
20775      * @private
20776      */
20777     getTargetCoord: function(iPageX, iPageY) {
20778
20779
20780         var x = iPageX - this.deltaX;
20781         var y = iPageY - this.deltaY;
20782
20783         if (this.constrainX) {
20784             if (x < this.minX) { x = this.minX; }
20785             if (x > this.maxX) { x = this.maxX; }
20786         }
20787
20788         if (this.constrainY) {
20789             if (y < this.minY) { y = this.minY; }
20790             if (y > this.maxY) { y = this.maxY; }
20791         }
20792
20793         x = this.getTick(x, this.xTicks);
20794         y = this.getTick(y, this.yTicks);
20795
20796
20797         return {x:x, y:y};
20798     },
20799
20800     /*
20801      * Sets up config options specific to this class. Overrides
20802      * Roo.dd.DragDrop, but all versions of this method through the
20803      * inheritance chain are called
20804      */
20805     applyConfig: function() {
20806         Roo.dd.DD.superclass.applyConfig.call(this);
20807         this.scroll = (this.config.scroll !== false);
20808     },
20809
20810     /*
20811      * Event that fires prior to the onMouseDown event.  Overrides
20812      * Roo.dd.DragDrop.
20813      */
20814     b4MouseDown: function(e) {
20815         // this.resetConstraints();
20816         this.autoOffset(e.getPageX(),
20817                             e.getPageY());
20818     },
20819
20820     /*
20821      * Event that fires prior to the onDrag event.  Overrides
20822      * Roo.dd.DragDrop.
20823      */
20824     b4Drag: function(e) {
20825         this.setDragElPos(e.getPageX(),
20826                             e.getPageY());
20827     },
20828
20829     toString: function() {
20830         return ("DD " + this.id);
20831     }
20832
20833     //////////////////////////////////////////////////////////////////////////
20834     // Debugging ygDragDrop events that can be overridden
20835     //////////////////////////////////////////////////////////////////////////
20836     /*
20837     startDrag: function(x, y) {
20838     },
20839
20840     onDrag: function(e) {
20841     },
20842
20843     onDragEnter: function(e, id) {
20844     },
20845
20846     onDragOver: function(e, id) {
20847     },
20848
20849     onDragOut: function(e, id) {
20850     },
20851
20852     onDragDrop: function(e, id) {
20853     },
20854
20855     endDrag: function(e) {
20856     }
20857
20858     */
20859
20860 });/*
20861  * Based on:
20862  * Ext JS Library 1.1.1
20863  * Copyright(c) 2006-2007, Ext JS, LLC.
20864  *
20865  * Originally Released Under LGPL - original licence link has changed is not relivant.
20866  *
20867  * Fork - LGPL
20868  * <script type="text/javascript">
20869  */
20870
20871 /**
20872  * @class Roo.dd.DDProxy
20873  * A DragDrop implementation that inserts an empty, bordered div into
20874  * the document that follows the cursor during drag operations.  At the time of
20875  * the click, the frame div is resized to the dimensions of the linked html
20876  * element, and moved to the exact location of the linked element.
20877  *
20878  * References to the "frame" element refer to the single proxy element that
20879  * was created to be dragged in place of all DDProxy elements on the
20880  * page.
20881  *
20882  * @extends Roo.dd.DD
20883  * @constructor
20884  * @param {String} id the id of the linked html element
20885  * @param {String} sGroup the group of related DragDrop objects
20886  * @param {object} config an object containing configurable attributes
20887  *                Valid properties for DDProxy in addition to those in DragDrop:
20888  *                   resizeFrame, centerFrame, dragElId
20889  */
20890 Roo.dd.DDProxy = function(id, sGroup, config) {
20891     if (id) {
20892         this.init(id, sGroup, config);
20893         this.initFrame();
20894     }
20895 };
20896
20897 /**
20898  * The default drag frame div id
20899  * @property Roo.dd.DDProxy.dragElId
20900  * @type String
20901  * @static
20902  */
20903 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20904
20905 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20906
20907     /**
20908      * By default we resize the drag frame to be the same size as the element
20909      * we want to drag (this is to get the frame effect).  We can turn it off
20910      * if we want a different behavior.
20911      * @property resizeFrame
20912      * @type boolean
20913      */
20914     resizeFrame: true,
20915
20916     /**
20917      * By default the frame is positioned exactly where the drag element is, so
20918      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20919      * you do not have constraints on the obj is to have the drag frame centered
20920      * around the cursor.  Set centerFrame to true for this effect.
20921      * @property centerFrame
20922      * @type boolean
20923      */
20924     centerFrame: false,
20925
20926     /**
20927      * Creates the proxy element if it does not yet exist
20928      * @method createFrame
20929      */
20930     createFrame: function() {
20931         var self = this;
20932         var body = document.body;
20933
20934         if (!body || !body.firstChild) {
20935             setTimeout( function() { self.createFrame(); }, 50 );
20936             return;
20937         }
20938
20939         var div = this.getDragEl();
20940
20941         if (!div) {
20942             div    = document.createElement("div");
20943             div.id = this.dragElId;
20944             var s  = div.style;
20945
20946             s.position   = "absolute";
20947             s.visibility = "hidden";
20948             s.cursor     = "move";
20949             s.border     = "2px solid #aaa";
20950             s.zIndex     = 999;
20951
20952             // appendChild can blow up IE if invoked prior to the window load event
20953             // while rendering a table.  It is possible there are other scenarios
20954             // that would cause this to happen as well.
20955             body.insertBefore(div, body.firstChild);
20956         }
20957     },
20958
20959     /**
20960      * Initialization for the drag frame element.  Must be called in the
20961      * constructor of all subclasses
20962      * @method initFrame
20963      */
20964     initFrame: function() {
20965         this.createFrame();
20966     },
20967
20968     applyConfig: function() {
20969         Roo.dd.DDProxy.superclass.applyConfig.call(this);
20970
20971         this.resizeFrame = (this.config.resizeFrame !== false);
20972         this.centerFrame = (this.config.centerFrame);
20973         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
20974     },
20975
20976     /**
20977      * Resizes the drag frame to the dimensions of the clicked object, positions
20978      * it over the object, and finally displays it
20979      * @method showFrame
20980      * @param {int} iPageX X click position
20981      * @param {int} iPageY Y click position
20982      * @private
20983      */
20984     showFrame: function(iPageX, iPageY) {
20985         var el = this.getEl();
20986         var dragEl = this.getDragEl();
20987         var s = dragEl.style;
20988
20989         this._resizeProxy();
20990
20991         if (this.centerFrame) {
20992             this.setDelta( Math.round(parseInt(s.width,  10)/2),
20993                            Math.round(parseInt(s.height, 10)/2) );
20994         }
20995
20996         this.setDragElPos(iPageX, iPageY);
20997
20998         Roo.fly(dragEl).show();
20999     },
21000
21001     /**
21002      * The proxy is automatically resized to the dimensions of the linked
21003      * element when a drag is initiated, unless resizeFrame is set to false
21004      * @method _resizeProxy
21005      * @private
21006      */
21007     _resizeProxy: function() {
21008         if (this.resizeFrame) {
21009             var el = this.getEl();
21010             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21011         }
21012     },
21013
21014     // overrides Roo.dd.DragDrop
21015     b4MouseDown: function(e) {
21016         var x = e.getPageX();
21017         var y = e.getPageY();
21018         this.autoOffset(x, y);
21019         this.setDragElPos(x, y);
21020     },
21021
21022     // overrides Roo.dd.DragDrop
21023     b4StartDrag: function(x, y) {
21024         // show the drag frame
21025         this.showFrame(x, y);
21026     },
21027
21028     // overrides Roo.dd.DragDrop
21029     b4EndDrag: function(e) {
21030         Roo.fly(this.getDragEl()).hide();
21031     },
21032
21033     // overrides Roo.dd.DragDrop
21034     // By default we try to move the element to the last location of the frame.
21035     // This is so that the default behavior mirrors that of Roo.dd.DD.
21036     endDrag: function(e) {
21037
21038         var lel = this.getEl();
21039         var del = this.getDragEl();
21040
21041         // Show the drag frame briefly so we can get its position
21042         del.style.visibility = "";
21043
21044         this.beforeMove();
21045         // Hide the linked element before the move to get around a Safari
21046         // rendering bug.
21047         lel.style.visibility = "hidden";
21048         Roo.dd.DDM.moveToEl(lel, del);
21049         del.style.visibility = "hidden";
21050         lel.style.visibility = "";
21051
21052         this.afterDrag();
21053     },
21054
21055     beforeMove : function(){
21056
21057     },
21058
21059     afterDrag : function(){
21060
21061     },
21062
21063     toString: function() {
21064         return ("DDProxy " + this.id);
21065     }
21066
21067 });
21068 /*
21069  * Based on:
21070  * Ext JS Library 1.1.1
21071  * Copyright(c) 2006-2007, Ext JS, LLC.
21072  *
21073  * Originally Released Under LGPL - original licence link has changed is not relivant.
21074  *
21075  * Fork - LGPL
21076  * <script type="text/javascript">
21077  */
21078
21079  /**
21080  * @class Roo.dd.DDTarget
21081  * A DragDrop implementation that does not move, but can be a drop
21082  * target.  You would get the same result by simply omitting implementation
21083  * for the event callbacks, but this way we reduce the processing cost of the
21084  * event listener and the callbacks.
21085  * @extends Roo.dd.DragDrop
21086  * @constructor
21087  * @param {String} id the id of the element that is a drop target
21088  * @param {String} sGroup the group of related DragDrop objects
21089  * @param {object} config an object containing configurable attributes
21090  *                 Valid properties for DDTarget in addition to those in
21091  *                 DragDrop:
21092  *                    none
21093  */
21094 Roo.dd.DDTarget = function(id, sGroup, config) {
21095     if (id) {
21096         this.initTarget(id, sGroup, config);
21097     }
21098     if (config.listeners || config.events) { 
21099        Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21100             listeners : config.listeners || {}, 
21101             events : config.events || {} 
21102         });    
21103     }
21104 };
21105
21106 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21107 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21108     toString: function() {
21109         return ("DDTarget " + this.id);
21110     }
21111 });
21112 /*
21113  * Based on:
21114  * Ext JS Library 1.1.1
21115  * Copyright(c) 2006-2007, Ext JS, LLC.
21116  *
21117  * Originally Released Under LGPL - original licence link has changed is not relivant.
21118  *
21119  * Fork - LGPL
21120  * <script type="text/javascript">
21121  */
21122  
21123
21124 /**
21125  * @class Roo.dd.ScrollManager
21126  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21127  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21128  * @singleton
21129  */
21130 Roo.dd.ScrollManager = function(){
21131     var ddm = Roo.dd.DragDropMgr;
21132     var els = {};
21133     var dragEl = null;
21134     var proc = {};
21135     
21136     
21137     
21138     var onStop = function(e){
21139         dragEl = null;
21140         clearProc();
21141     };
21142     
21143     var triggerRefresh = function(){
21144         if(ddm.dragCurrent){
21145              ddm.refreshCache(ddm.dragCurrent.groups);
21146         }
21147     };
21148     
21149     var doScroll = function(){
21150         if(ddm.dragCurrent){
21151             var dds = Roo.dd.ScrollManager;
21152             if(!dds.animate){
21153                 if(proc.el.scroll(proc.dir, dds.increment)){
21154                     triggerRefresh();
21155                 }
21156             }else{
21157                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21158             }
21159         }
21160     };
21161     
21162     var clearProc = function(){
21163         if(proc.id){
21164             clearInterval(proc.id);
21165         }
21166         proc.id = 0;
21167         proc.el = null;
21168         proc.dir = "";
21169     };
21170     
21171     var startProc = function(el, dir){
21172          Roo.log('scroll startproc');
21173         clearProc();
21174         proc.el = el;
21175         proc.dir = dir;
21176         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21177     };
21178     
21179     var onFire = function(e, isDrop){
21180        
21181         if(isDrop || !ddm.dragCurrent){ return; }
21182         var dds = Roo.dd.ScrollManager;
21183         if(!dragEl || dragEl != ddm.dragCurrent){
21184             dragEl = ddm.dragCurrent;
21185             // refresh regions on drag start
21186             dds.refreshCache();
21187         }
21188         
21189         var xy = Roo.lib.Event.getXY(e);
21190         var pt = new Roo.lib.Point(xy[0], xy[1]);
21191         for(var id in els){
21192             var el = els[id], r = el._region;
21193             if(r && r.contains(pt) && el.isScrollable()){
21194                 if(r.bottom - pt.y <= dds.thresh){
21195                     if(proc.el != el){
21196                         startProc(el, "down");
21197                     }
21198                     return;
21199                 }else if(r.right - pt.x <= dds.thresh){
21200                     if(proc.el != el){
21201                         startProc(el, "left");
21202                     }
21203                     return;
21204                 }else if(pt.y - r.top <= dds.thresh){
21205                     if(proc.el != el){
21206                         startProc(el, "up");
21207                     }
21208                     return;
21209                 }else if(pt.x - r.left <= dds.thresh){
21210                     if(proc.el != el){
21211                         startProc(el, "right");
21212                     }
21213                     return;
21214                 }
21215             }
21216         }
21217         clearProc();
21218     };
21219     
21220     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21221     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21222     
21223     return {
21224         /**
21225          * Registers new overflow element(s) to auto scroll
21226          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21227          */
21228         register : function(el){
21229             if(el instanceof Array){
21230                 for(var i = 0, len = el.length; i < len; i++) {
21231                         this.register(el[i]);
21232                 }
21233             }else{
21234                 el = Roo.get(el);
21235                 els[el.id] = el;
21236             }
21237             Roo.dd.ScrollManager.els = els;
21238         },
21239         
21240         /**
21241          * Unregisters overflow element(s) so they are no longer scrolled
21242          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21243          */
21244         unregister : function(el){
21245             if(el instanceof Array){
21246                 for(var i = 0, len = el.length; i < len; i++) {
21247                         this.unregister(el[i]);
21248                 }
21249             }else{
21250                 el = Roo.get(el);
21251                 delete els[el.id];
21252             }
21253         },
21254         
21255         /**
21256          * The number of pixels from the edge of a container the pointer needs to be to 
21257          * trigger scrolling (defaults to 25)
21258          * @type Number
21259          */
21260         thresh : 25,
21261         
21262         /**
21263          * The number of pixels to scroll in each scroll increment (defaults to 50)
21264          * @type Number
21265          */
21266         increment : 100,
21267         
21268         /**
21269          * The frequency of scrolls in milliseconds (defaults to 500)
21270          * @type Number
21271          */
21272         frequency : 500,
21273         
21274         /**
21275          * True to animate the scroll (defaults to true)
21276          * @type Boolean
21277          */
21278         animate: true,
21279         
21280         /**
21281          * The animation duration in seconds - 
21282          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21283          * @type Number
21284          */
21285         animDuration: .4,
21286         
21287         /**
21288          * Manually trigger a cache refresh.
21289          */
21290         refreshCache : function(){
21291             for(var id in els){
21292                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21293                     els[id]._region = els[id].getRegion();
21294                 }
21295             }
21296         }
21297     };
21298 }();/*
21299  * Based on:
21300  * Ext JS Library 1.1.1
21301  * Copyright(c) 2006-2007, Ext JS, LLC.
21302  *
21303  * Originally Released Under LGPL - original licence link has changed is not relivant.
21304  *
21305  * Fork - LGPL
21306  * <script type="text/javascript">
21307  */
21308  
21309
21310 /**
21311  * @class Roo.dd.Registry
21312  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21313  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21314  * @singleton
21315  */
21316 Roo.dd.Registry = function(){
21317     var elements = {}; 
21318     var handles = {}; 
21319     var autoIdSeed = 0;
21320
21321     var getId = function(el, autogen){
21322         if(typeof el == "string"){
21323             return el;
21324         }
21325         var id = el.id;
21326         if(!id && autogen !== false){
21327             id = "roodd-" + (++autoIdSeed);
21328             el.id = id;
21329         }
21330         return id;
21331     };
21332     
21333     return {
21334     /**
21335      * Register a drag drop element
21336      * @param {String|HTMLElement} element The id or DOM node to register
21337      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21338      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21339      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21340      * populated in the data object (if applicable):
21341      * <pre>
21342 Value      Description<br />
21343 ---------  ------------------------------------------<br />
21344 handles    Array of DOM nodes that trigger dragging<br />
21345            for the element being registered<br />
21346 isHandle   True if the element passed in triggers<br />
21347            dragging itself, else false
21348 </pre>
21349      */
21350         register : function(el, data){
21351             data = data || {};
21352             if(typeof el == "string"){
21353                 el = document.getElementById(el);
21354             }
21355             data.ddel = el;
21356             elements[getId(el)] = data;
21357             if(data.isHandle !== false){
21358                 handles[data.ddel.id] = data;
21359             }
21360             if(data.handles){
21361                 var hs = data.handles;
21362                 for(var i = 0, len = hs.length; i < len; i++){
21363                         handles[getId(hs[i])] = data;
21364                 }
21365             }
21366         },
21367
21368     /**
21369      * Unregister a drag drop element
21370      * @param {String|HTMLElement}  element The id or DOM node to unregister
21371      */
21372         unregister : function(el){
21373             var id = getId(el, false);
21374             var data = elements[id];
21375             if(data){
21376                 delete elements[id];
21377                 if(data.handles){
21378                     var hs = data.handles;
21379                     for(var i = 0, len = hs.length; i < len; i++){
21380                         delete handles[getId(hs[i], false)];
21381                     }
21382                 }
21383             }
21384         },
21385
21386     /**
21387      * Returns the handle registered for a DOM Node by id
21388      * @param {String|HTMLElement} id The DOM node or id to look up
21389      * @return {Object} handle The custom handle data
21390      */
21391         getHandle : function(id){
21392             if(typeof id != "string"){ // must be element?
21393                 id = id.id;
21394             }
21395             return handles[id];
21396         },
21397
21398     /**
21399      * Returns the handle that is registered for the DOM node that is the target of the event
21400      * @param {Event} e The event
21401      * @return {Object} handle The custom handle data
21402      */
21403         getHandleFromEvent : function(e){
21404             var t = Roo.lib.Event.getTarget(e);
21405             return t ? handles[t.id] : null;
21406         },
21407
21408     /**
21409      * Returns a custom data object that is registered for a DOM node by id
21410      * @param {String|HTMLElement} id The DOM node or id to look up
21411      * @return {Object} data The custom data
21412      */
21413         getTarget : function(id){
21414             if(typeof id != "string"){ // must be element?
21415                 id = id.id;
21416             }
21417             return elements[id];
21418         },
21419
21420     /**
21421      * Returns a custom data object that is registered for the DOM node that is the target of the event
21422      * @param {Event} e The event
21423      * @return {Object} data The custom data
21424      */
21425         getTargetFromEvent : function(e){
21426             var t = Roo.lib.Event.getTarget(e);
21427             return t ? elements[t.id] || handles[t.id] : null;
21428         }
21429     };
21430 }();/*
21431  * Based on:
21432  * Ext JS Library 1.1.1
21433  * Copyright(c) 2006-2007, Ext JS, LLC.
21434  *
21435  * Originally Released Under LGPL - original licence link has changed is not relivant.
21436  *
21437  * Fork - LGPL
21438  * <script type="text/javascript">
21439  */
21440  
21441
21442 /**
21443  * @class Roo.dd.StatusProxy
21444  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21445  * default drag proxy used by all Roo.dd components.
21446  * @constructor
21447  * @param {Object} config
21448  */
21449 Roo.dd.StatusProxy = function(config){
21450     Roo.apply(this, config);
21451     this.id = this.id || Roo.id();
21452     this.el = new Roo.Layer({
21453         dh: {
21454             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21455                 {tag: "div", cls: "x-dd-drop-icon"},
21456                 {tag: "div", cls: "x-dd-drag-ghost"}
21457             ]
21458         }, 
21459         shadow: !config || config.shadow !== false
21460     });
21461     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21462     this.dropStatus = this.dropNotAllowed;
21463 };
21464
21465 Roo.dd.StatusProxy.prototype = {
21466     /**
21467      * @cfg {String} dropAllowed
21468      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21469      */
21470     dropAllowed : "x-dd-drop-ok",
21471     /**
21472      * @cfg {String} dropNotAllowed
21473      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21474      */
21475     dropNotAllowed : "x-dd-drop-nodrop",
21476
21477     /**
21478      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21479      * over the current target element.
21480      * @param {String} cssClass The css class for the new drop status indicator image
21481      */
21482     setStatus : function(cssClass){
21483         cssClass = cssClass || this.dropNotAllowed;
21484         if(this.dropStatus != cssClass){
21485             this.el.replaceClass(this.dropStatus, cssClass);
21486             this.dropStatus = cssClass;
21487         }
21488     },
21489
21490     /**
21491      * Resets the status indicator to the default dropNotAllowed value
21492      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21493      */
21494     reset : function(clearGhost){
21495         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21496         this.dropStatus = this.dropNotAllowed;
21497         if(clearGhost){
21498             this.ghost.update("");
21499         }
21500     },
21501
21502     /**
21503      * Updates the contents of the ghost element
21504      * @param {String} html The html that will replace the current innerHTML of the ghost element
21505      */
21506     update : function(html){
21507         if(typeof html == "string"){
21508             this.ghost.update(html);
21509         }else{
21510             this.ghost.update("");
21511             html.style.margin = "0";
21512             this.ghost.dom.appendChild(html);
21513         }
21514         // ensure float = none set?? cant remember why though.
21515         var el = this.ghost.dom.firstChild;
21516                 if(el){
21517                         Roo.fly(el).setStyle('float', 'none');
21518                 }
21519     },
21520     
21521     /**
21522      * Returns the underlying proxy {@link Roo.Layer}
21523      * @return {Roo.Layer} el
21524     */
21525     getEl : function(){
21526         return this.el;
21527     },
21528
21529     /**
21530      * Returns the ghost element
21531      * @return {Roo.Element} el
21532      */
21533     getGhost : function(){
21534         return this.ghost;
21535     },
21536
21537     /**
21538      * Hides the proxy
21539      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21540      */
21541     hide : function(clear){
21542         this.el.hide();
21543         if(clear){
21544             this.reset(true);
21545         }
21546     },
21547
21548     /**
21549      * Stops the repair animation if it's currently running
21550      */
21551     stop : function(){
21552         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21553             this.anim.stop();
21554         }
21555     },
21556
21557     /**
21558      * Displays this proxy
21559      */
21560     show : function(){
21561         this.el.show();
21562     },
21563
21564     /**
21565      * Force the Layer to sync its shadow and shim positions to the element
21566      */
21567     sync : function(){
21568         this.el.sync();
21569     },
21570
21571     /**
21572      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21573      * invalid drop operation by the item being dragged.
21574      * @param {Array} xy The XY position of the element ([x, y])
21575      * @param {Function} callback The function to call after the repair is complete
21576      * @param {Object} scope The scope in which to execute the callback
21577      */
21578     repair : function(xy, callback, scope){
21579         this.callback = callback;
21580         this.scope = scope;
21581         if(xy && this.animRepair !== false){
21582             this.el.addClass("x-dd-drag-repair");
21583             this.el.hideUnders(true);
21584             this.anim = this.el.shift({
21585                 duration: this.repairDuration || .5,
21586                 easing: 'easeOut',
21587                 xy: xy,
21588                 stopFx: true,
21589                 callback: this.afterRepair,
21590                 scope: this
21591             });
21592         }else{
21593             this.afterRepair();
21594         }
21595     },
21596
21597     // private
21598     afterRepair : function(){
21599         this.hide(true);
21600         if(typeof this.callback == "function"){
21601             this.callback.call(this.scope || this);
21602         }
21603         this.callback = null;
21604         this.scope = null;
21605     }
21606 };/*
21607  * Based on:
21608  * Ext JS Library 1.1.1
21609  * Copyright(c) 2006-2007, Ext JS, LLC.
21610  *
21611  * Originally Released Under LGPL - original licence link has changed is not relivant.
21612  *
21613  * Fork - LGPL
21614  * <script type="text/javascript">
21615  */
21616
21617 /**
21618  * @class Roo.dd.DragSource
21619  * @extends Roo.dd.DDProxy
21620  * A simple class that provides the basic implementation needed to make any element draggable.
21621  * @constructor
21622  * @param {String/HTMLElement/Element} el The container element
21623  * @param {Object} config
21624  */
21625 Roo.dd.DragSource = function(el, config){
21626     this.el = Roo.get(el);
21627     this.dragData = {};
21628     
21629     Roo.apply(this, config);
21630     
21631     if(!this.proxy){
21632         this.proxy = new Roo.dd.StatusProxy();
21633     }
21634
21635     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21636           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21637     
21638     this.dragging = false;
21639 };
21640
21641 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21642     /**
21643      * @cfg {String} dropAllowed
21644      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21645      */
21646     dropAllowed : "x-dd-drop-ok",
21647     /**
21648      * @cfg {String} dropNotAllowed
21649      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21650      */
21651     dropNotAllowed : "x-dd-drop-nodrop",
21652
21653     /**
21654      * Returns the data object associated with this drag source
21655      * @return {Object} data An object containing arbitrary data
21656      */
21657     getDragData : function(e){
21658         return this.dragData;
21659     },
21660
21661     // private
21662     onDragEnter : function(e, id){
21663         var target = Roo.dd.DragDropMgr.getDDById(id);
21664         this.cachedTarget = target;
21665         if(this.beforeDragEnter(target, e, id) !== false){
21666             if(target.isNotifyTarget){
21667                 var status = target.notifyEnter(this, e, this.dragData);
21668                 this.proxy.setStatus(status);
21669             }else{
21670                 this.proxy.setStatus(this.dropAllowed);
21671             }
21672             
21673             if(this.afterDragEnter){
21674                 /**
21675                  * An empty function by default, but provided so that you can perform a custom action
21676                  * when the dragged item enters the drop target by providing an implementation.
21677                  * @param {Roo.dd.DragDrop} target The drop target
21678                  * @param {Event} e The event object
21679                  * @param {String} id The id of the dragged element
21680                  * @method afterDragEnter
21681                  */
21682                 this.afterDragEnter(target, e, id);
21683             }
21684         }
21685     },
21686
21687     /**
21688      * An empty function by default, but provided so that you can perform a custom action
21689      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21690      * @param {Roo.dd.DragDrop} target The drop target
21691      * @param {Event} e The event object
21692      * @param {String} id The id of the dragged element
21693      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21694      */
21695     beforeDragEnter : function(target, e, id){
21696         return true;
21697     },
21698
21699     // private
21700     alignElWithMouse: function() {
21701         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21702         this.proxy.sync();
21703     },
21704
21705     // private
21706     onDragOver : function(e, id){
21707         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21708         if(this.beforeDragOver(target, e, id) !== false){
21709             if(target.isNotifyTarget){
21710                 var status = target.notifyOver(this, e, this.dragData);
21711                 this.proxy.setStatus(status);
21712             }
21713
21714             if(this.afterDragOver){
21715                 /**
21716                  * An empty function by default, but provided so that you can perform a custom action
21717                  * while the dragged item is over the drop target by providing an implementation.
21718                  * @param {Roo.dd.DragDrop} target The drop target
21719                  * @param {Event} e The event object
21720                  * @param {String} id The id of the dragged element
21721                  * @method afterDragOver
21722                  */
21723                 this.afterDragOver(target, e, id);
21724             }
21725         }
21726     },
21727
21728     /**
21729      * An empty function by default, but provided so that you can perform a custom action
21730      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21731      * @param {Roo.dd.DragDrop} target The drop target
21732      * @param {Event} e The event object
21733      * @param {String} id The id of the dragged element
21734      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21735      */
21736     beforeDragOver : function(target, e, id){
21737         return true;
21738     },
21739
21740     // private
21741     onDragOut : function(e, id){
21742         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21743         if(this.beforeDragOut(target, e, id) !== false){
21744             if(target.isNotifyTarget){
21745                 target.notifyOut(this, e, this.dragData);
21746             }
21747             this.proxy.reset();
21748             if(this.afterDragOut){
21749                 /**
21750                  * An empty function by default, but provided so that you can perform a custom action
21751                  * after the dragged item is dragged out of the target without dropping.
21752                  * @param {Roo.dd.DragDrop} target The drop target
21753                  * @param {Event} e The event object
21754                  * @param {String} id The id of the dragged element
21755                  * @method afterDragOut
21756                  */
21757                 this.afterDragOut(target, e, id);
21758             }
21759         }
21760         this.cachedTarget = null;
21761     },
21762
21763     /**
21764      * An empty function by default, but provided so that you can perform a custom action before the dragged
21765      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21766      * @param {Roo.dd.DragDrop} target The drop target
21767      * @param {Event} e The event object
21768      * @param {String} id The id of the dragged element
21769      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21770      */
21771     beforeDragOut : function(target, e, id){
21772         return true;
21773     },
21774     
21775     // private
21776     onDragDrop : function(e, id){
21777         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21778         if(this.beforeDragDrop(target, e, id) !== false){
21779             if(target.isNotifyTarget){
21780                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21781                     this.onValidDrop(target, e, id);
21782                 }else{
21783                     this.onInvalidDrop(target, e, id);
21784                 }
21785             }else{
21786                 this.onValidDrop(target, e, id);
21787             }
21788             
21789             if(this.afterDragDrop){
21790                 /**
21791                  * An empty function by default, but provided so that you can perform a custom action
21792                  * after a valid drag drop has occurred by providing an implementation.
21793                  * @param {Roo.dd.DragDrop} target The drop target
21794                  * @param {Event} e The event object
21795                  * @param {String} id The id of the dropped element
21796                  * @method afterDragDrop
21797                  */
21798                 this.afterDragDrop(target, e, id);
21799             }
21800         }
21801         delete this.cachedTarget;
21802     },
21803
21804     /**
21805      * An empty function by default, but provided so that you can perform a custom action before the dragged
21806      * item is dropped onto the target and optionally cancel the onDragDrop.
21807      * @param {Roo.dd.DragDrop} target The drop target
21808      * @param {Event} e The event object
21809      * @param {String} id The id of the dragged element
21810      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21811      */
21812     beforeDragDrop : function(target, e, id){
21813         return true;
21814     },
21815
21816     // private
21817     onValidDrop : function(target, e, id){
21818         this.hideProxy();
21819         if(this.afterValidDrop){
21820             /**
21821              * An empty function by default, but provided so that you can perform a custom action
21822              * after a valid drop has occurred by providing an implementation.
21823              * @param {Object} target The target DD 
21824              * @param {Event} e The event object
21825              * @param {String} id The id of the dropped element
21826              * @method afterInvalidDrop
21827              */
21828             this.afterValidDrop(target, e, id);
21829         }
21830     },
21831
21832     // private
21833     getRepairXY : function(e, data){
21834         return this.el.getXY();  
21835     },
21836
21837     // private
21838     onInvalidDrop : function(target, e, id){
21839         this.beforeInvalidDrop(target, e, id);
21840         if(this.cachedTarget){
21841             if(this.cachedTarget.isNotifyTarget){
21842                 this.cachedTarget.notifyOut(this, e, this.dragData);
21843             }
21844             this.cacheTarget = null;
21845         }
21846         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21847
21848         if(this.afterInvalidDrop){
21849             /**
21850              * An empty function by default, but provided so that you can perform a custom action
21851              * after an invalid drop has occurred by providing an implementation.
21852              * @param {Event} e The event object
21853              * @param {String} id The id of the dropped element
21854              * @method afterInvalidDrop
21855              */
21856             this.afterInvalidDrop(e, id);
21857         }
21858     },
21859
21860     // private
21861     afterRepair : function(){
21862         if(Roo.enableFx){
21863             this.el.highlight(this.hlColor || "c3daf9");
21864         }
21865         this.dragging = false;
21866     },
21867
21868     /**
21869      * An empty function by default, but provided so that you can perform a custom action after an invalid
21870      * drop has occurred.
21871      * @param {Roo.dd.DragDrop} target The drop target
21872      * @param {Event} e The event object
21873      * @param {String} id The id of the dragged element
21874      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21875      */
21876     beforeInvalidDrop : function(target, e, id){
21877         return true;
21878     },
21879
21880     // private
21881     handleMouseDown : function(e){
21882         if(this.dragging) {
21883             return;
21884         }
21885         var data = this.getDragData(e);
21886         if(data && this.onBeforeDrag(data, e) !== false){
21887             this.dragData = data;
21888             this.proxy.stop();
21889             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21890         } 
21891     },
21892
21893     /**
21894      * An empty function by default, but provided so that you can perform a custom action before the initial
21895      * drag event begins and optionally cancel it.
21896      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21897      * @param {Event} e The event object
21898      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21899      */
21900     onBeforeDrag : function(data, e){
21901         return true;
21902     },
21903
21904     /**
21905      * An empty function by default, but provided so that you can perform a custom action once the initial
21906      * drag event has begun.  The drag cannot be canceled from this function.
21907      * @param {Number} x The x position of the click on the dragged object
21908      * @param {Number} y The y position of the click on the dragged object
21909      */
21910     onStartDrag : Roo.emptyFn,
21911
21912     // private - YUI override
21913     startDrag : function(x, y){
21914         this.proxy.reset();
21915         this.dragging = true;
21916         this.proxy.update("");
21917         this.onInitDrag(x, y);
21918         this.proxy.show();
21919     },
21920
21921     // private
21922     onInitDrag : function(x, y){
21923         var clone = this.el.dom.cloneNode(true);
21924         clone.id = Roo.id(); // prevent duplicate ids
21925         this.proxy.update(clone);
21926         this.onStartDrag(x, y);
21927         return true;
21928     },
21929
21930     /**
21931      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21932      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21933      */
21934     getProxy : function(){
21935         return this.proxy;  
21936     },
21937
21938     /**
21939      * Hides the drag source's {@link Roo.dd.StatusProxy}
21940      */
21941     hideProxy : function(){
21942         this.proxy.hide();  
21943         this.proxy.reset(true);
21944         this.dragging = false;
21945     },
21946
21947     // private
21948     triggerCacheRefresh : function(){
21949         Roo.dd.DDM.refreshCache(this.groups);
21950     },
21951
21952     // private - override to prevent hiding
21953     b4EndDrag: function(e) {
21954     },
21955
21956     // private - override to prevent moving
21957     endDrag : function(e){
21958         this.onEndDrag(this.dragData, e);
21959     },
21960
21961     // private
21962     onEndDrag : function(data, e){
21963     },
21964     
21965     // private - pin to cursor
21966     autoOffset : function(x, y) {
21967         this.setDelta(-12, -20);
21968     }    
21969 });/*
21970  * Based on:
21971  * Ext JS Library 1.1.1
21972  * Copyright(c) 2006-2007, Ext JS, LLC.
21973  *
21974  * Originally Released Under LGPL - original licence link has changed is not relivant.
21975  *
21976  * Fork - LGPL
21977  * <script type="text/javascript">
21978  */
21979
21980
21981 /**
21982  * @class Roo.dd.DropTarget
21983  * @extends Roo.dd.DDTarget
21984  * A simple class that provides the basic implementation needed to make any element a drop target that can have
21985  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
21986  * @constructor
21987  * @param {String/HTMLElement/Element} el The container element
21988  * @param {Object} config
21989  */
21990 Roo.dd.DropTarget = function(el, config){
21991     this.el = Roo.get(el);
21992     
21993     var listeners = false; ;
21994     if (config && config.listeners) {
21995         listeners= config.listeners;
21996         delete config.listeners;
21997     }
21998     Roo.apply(this, config);
21999     
22000     if(this.containerScroll){
22001         Roo.dd.ScrollManager.register(this.el);
22002     }
22003     this.addEvents( {
22004          /**
22005          * @scope Roo.dd.DropTarget
22006          */
22007          
22008          /**
22009          * @event enter
22010          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22011          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22012          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22013          * 
22014          * IMPORTANT : it should set this.overClass and this.dropAllowed
22015          * 
22016          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22017          * @param {Event} e The event
22018          * @param {Object} data An object containing arbitrary data supplied by the drag source
22019          */
22020         "enter" : true,
22021         
22022          /**
22023          * @event over
22024          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22025          * This method will be called on every mouse movement while the drag source is over the drop target.
22026          * This default implementation simply returns the dropAllowed config value.
22027          * 
22028          * IMPORTANT : it should set this.dropAllowed
22029          * 
22030          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22031          * @param {Event} e The event
22032          * @param {Object} data An object containing arbitrary data supplied by the drag source
22033          
22034          */
22035         "over" : true,
22036         /**
22037          * @event out
22038          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22039          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22040          * overClass (if any) from the drop element.
22041          * 
22042          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22043          * @param {Event} e The event
22044          * @param {Object} data An object containing arbitrary data supplied by the drag source
22045          */
22046          "out" : true,
22047          
22048         /**
22049          * @event drop
22050          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22051          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22052          * implementation that does something to process the drop event and returns true so that the drag source's
22053          * repair action does not run.
22054          * 
22055          * IMPORTANT : it should set this.success
22056          * 
22057          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22058          * @param {Event} e The event
22059          * @param {Object} data An object containing arbitrary data supplied by the drag source
22060         */
22061          "drop" : true
22062     });
22063             
22064      
22065     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22066         this.el.dom, 
22067         this.ddGroup || this.group,
22068         {
22069             isTarget: true,
22070             listeners : listeners || {} 
22071            
22072         
22073         }
22074     );
22075
22076 };
22077
22078 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22079     /**
22080      * @cfg {String} overClass
22081      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22082      */
22083      /**
22084      * @cfg {String} ddGroup
22085      * The drag drop group to handle drop events for
22086      */
22087      
22088     /**
22089      * @cfg {String} dropAllowed
22090      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22091      */
22092     dropAllowed : "x-dd-drop-ok",
22093     /**
22094      * @cfg {String} dropNotAllowed
22095      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22096      */
22097     dropNotAllowed : "x-dd-drop-nodrop",
22098     /**
22099      * @cfg {boolean} success
22100      * set this after drop listener.. 
22101      */
22102     success : false,
22103     /**
22104      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22105      * if the drop point is valid for over/enter..
22106      */
22107     valid : false,
22108     // private
22109     isTarget : true,
22110
22111     // private
22112     isNotifyTarget : true,
22113     
22114     /**
22115      * @hide
22116      */
22117     notifyEnter : function(dd, e, data)
22118     {
22119         this.valid = true;
22120         this.fireEvent('enter', dd, e, data);
22121         if(this.overClass){
22122             this.el.addClass(this.overClass);
22123         }
22124         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22125             this.valid ? this.dropAllowed : this.dropNotAllowed
22126         );
22127     },
22128
22129     /**
22130      * @hide
22131      */
22132     notifyOver : function(dd, e, data)
22133     {
22134         this.valid = true;
22135         this.fireEvent('over', dd, e, data);
22136         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22137             this.valid ? this.dropAllowed : this.dropNotAllowed
22138         );
22139     },
22140
22141     /**
22142      * @hide
22143      */
22144     notifyOut : function(dd, e, data)
22145     {
22146         this.fireEvent('out', dd, e, data);
22147         if(this.overClass){
22148             this.el.removeClass(this.overClass);
22149         }
22150     },
22151
22152     /**
22153      * @hide
22154      */
22155     notifyDrop : function(dd, e, data)
22156     {
22157         this.success = false;
22158         this.fireEvent('drop', dd, e, data);
22159         return this.success;
22160     }
22161 });/*
22162  * Based on:
22163  * Ext JS Library 1.1.1
22164  * Copyright(c) 2006-2007, Ext JS, LLC.
22165  *
22166  * Originally Released Under LGPL - original licence link has changed is not relivant.
22167  *
22168  * Fork - LGPL
22169  * <script type="text/javascript">
22170  */
22171
22172
22173 /**
22174  * @class Roo.dd.DragZone
22175  * @extends Roo.dd.DragSource
22176  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22177  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22178  * @constructor
22179  * @param {String/HTMLElement/Element} el The container element
22180  * @param {Object} config
22181  */
22182 Roo.dd.DragZone = function(el, config){
22183     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22184     if(this.containerScroll){
22185         Roo.dd.ScrollManager.register(this.el);
22186     }
22187 };
22188
22189 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22190     /**
22191      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22192      * for auto scrolling during drag operations.
22193      */
22194     /**
22195      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22196      * method after a failed drop (defaults to "c3daf9" - light blue)
22197      */
22198
22199     /**
22200      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22201      * for a valid target to drag based on the mouse down. Override this method
22202      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22203      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22204      * @param {EventObject} e The mouse down event
22205      * @return {Object} The dragData
22206      */
22207     getDragData : function(e){
22208         return Roo.dd.Registry.getHandleFromEvent(e);
22209     },
22210     
22211     /**
22212      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22213      * this.dragData.ddel
22214      * @param {Number} x The x position of the click on the dragged object
22215      * @param {Number} y The y position of the click on the dragged object
22216      * @return {Boolean} true to continue the drag, false to cancel
22217      */
22218     onInitDrag : function(x, y){
22219         this.proxy.update(this.dragData.ddel.cloneNode(true));
22220         this.onStartDrag(x, y);
22221         return true;
22222     },
22223     
22224     /**
22225      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22226      */
22227     afterRepair : function(){
22228         if(Roo.enableFx){
22229             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22230         }
22231         this.dragging = false;
22232     },
22233
22234     /**
22235      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22236      * the XY of this.dragData.ddel
22237      * @param {EventObject} e The mouse up event
22238      * @return {Array} The xy location (e.g. [100, 200])
22239      */
22240     getRepairXY : function(e){
22241         return Roo.Element.fly(this.dragData.ddel).getXY();  
22242     }
22243 });/*
22244  * Based on:
22245  * Ext JS Library 1.1.1
22246  * Copyright(c) 2006-2007, Ext JS, LLC.
22247  *
22248  * Originally Released Under LGPL - original licence link has changed is not relivant.
22249  *
22250  * Fork - LGPL
22251  * <script type="text/javascript">
22252  */
22253 /**
22254  * @class Roo.dd.DropZone
22255  * @extends Roo.dd.DropTarget
22256  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22257  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22258  * @constructor
22259  * @param {String/HTMLElement/Element} el The container element
22260  * @param {Object} config
22261  */
22262 Roo.dd.DropZone = function(el, config){
22263     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22264 };
22265
22266 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22267     /**
22268      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22269      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22270      * provide your own custom lookup.
22271      * @param {Event} e The event
22272      * @return {Object} data The custom data
22273      */
22274     getTargetFromEvent : function(e){
22275         return Roo.dd.Registry.getTargetFromEvent(e);
22276     },
22277
22278     /**
22279      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22280      * that it has registered.  This method has no default implementation and should be overridden to provide
22281      * node-specific processing if necessary.
22282      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22283      * {@link #getTargetFromEvent} for this node)
22284      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22285      * @param {Event} e The event
22286      * @param {Object} data An object containing arbitrary data supplied by the drag source
22287      */
22288     onNodeEnter : function(n, dd, e, data){
22289         
22290     },
22291
22292     /**
22293      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22294      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22295      * overridden to provide the proper feedback.
22296      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22297      * {@link #getTargetFromEvent} for this node)
22298      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22299      * @param {Event} e The event
22300      * @param {Object} data An object containing arbitrary data supplied by the drag source
22301      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22302      * underlying {@link Roo.dd.StatusProxy} can be updated
22303      */
22304     onNodeOver : function(n, dd, e, data){
22305         return this.dropAllowed;
22306     },
22307
22308     /**
22309      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22310      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22311      * node-specific processing if necessary.
22312      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22313      * {@link #getTargetFromEvent} for this node)
22314      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22315      * @param {Event} e The event
22316      * @param {Object} data An object containing arbitrary data supplied by the drag source
22317      */
22318     onNodeOut : function(n, dd, e, data){
22319         
22320     },
22321
22322     /**
22323      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22324      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22325      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22326      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22327      * {@link #getTargetFromEvent} for this node)
22328      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22329      * @param {Event} e The event
22330      * @param {Object} data An object containing arbitrary data supplied by the drag source
22331      * @return {Boolean} True if the drop was valid, else false
22332      */
22333     onNodeDrop : function(n, dd, e, data){
22334         return false;
22335     },
22336
22337     /**
22338      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22339      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22340      * it should be overridden to provide the proper feedback if necessary.
22341      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22342      * @param {Event} e The event
22343      * @param {Object} data An object containing arbitrary data supplied by the drag source
22344      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22345      * underlying {@link Roo.dd.StatusProxy} can be updated
22346      */
22347     onContainerOver : function(dd, e, data){
22348         return this.dropNotAllowed;
22349     },
22350
22351     /**
22352      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22353      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22354      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22355      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22356      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22357      * @param {Event} e The event
22358      * @param {Object} data An object containing arbitrary data supplied by the drag source
22359      * @return {Boolean} True if the drop was valid, else false
22360      */
22361     onContainerDrop : function(dd, e, data){
22362         return false;
22363     },
22364
22365     /**
22366      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22367      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22368      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22369      * you should override this method and provide a custom implementation.
22370      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22371      * @param {Event} e The event
22372      * @param {Object} data An object containing arbitrary data supplied by the drag source
22373      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22374      * underlying {@link Roo.dd.StatusProxy} can be updated
22375      */
22376     notifyEnter : function(dd, e, data){
22377         return this.dropNotAllowed;
22378     },
22379
22380     /**
22381      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22382      * This method will be called on every mouse movement while the drag source is over the drop zone.
22383      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22384      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22385      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22386      * registered node, it will call {@link #onContainerOver}.
22387      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22388      * @param {Event} e The event
22389      * @param {Object} data An object containing arbitrary data supplied by the drag source
22390      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22391      * underlying {@link Roo.dd.StatusProxy} can be updated
22392      */
22393     notifyOver : function(dd, e, data){
22394         var n = this.getTargetFromEvent(e);
22395         if(!n){ // not over valid drop target
22396             if(this.lastOverNode){
22397                 this.onNodeOut(this.lastOverNode, dd, e, data);
22398                 this.lastOverNode = null;
22399             }
22400             return this.onContainerOver(dd, e, data);
22401         }
22402         if(this.lastOverNode != n){
22403             if(this.lastOverNode){
22404                 this.onNodeOut(this.lastOverNode, dd, e, data);
22405             }
22406             this.onNodeEnter(n, dd, e, data);
22407             this.lastOverNode = n;
22408         }
22409         return this.onNodeOver(n, dd, e, data);
22410     },
22411
22412     /**
22413      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22414      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22415      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22416      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22417      * @param {Event} e The event
22418      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22419      */
22420     notifyOut : function(dd, e, data){
22421         if(this.lastOverNode){
22422             this.onNodeOut(this.lastOverNode, dd, e, data);
22423             this.lastOverNode = null;
22424         }
22425     },
22426
22427     /**
22428      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22429      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22430      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22431      * otherwise it will call {@link #onContainerDrop}.
22432      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22433      * @param {Event} e The event
22434      * @param {Object} data An object containing arbitrary data supplied by the drag source
22435      * @return {Boolean} True if the drop was valid, else false
22436      */
22437     notifyDrop : function(dd, e, data){
22438         if(this.lastOverNode){
22439             this.onNodeOut(this.lastOverNode, dd, e, data);
22440             this.lastOverNode = null;
22441         }
22442         var n = this.getTargetFromEvent(e);
22443         return n ?
22444             this.onNodeDrop(n, dd, e, data) :
22445             this.onContainerDrop(dd, e, data);
22446     },
22447
22448     // private
22449     triggerCacheRefresh : function(){
22450         Roo.dd.DDM.refreshCache(this.groups);
22451     }  
22452 });/*
22453  * Based on:
22454  * Ext JS Library 1.1.1
22455  * Copyright(c) 2006-2007, Ext JS, LLC.
22456  *
22457  * Originally Released Under LGPL - original licence link has changed is not relivant.
22458  *
22459  * Fork - LGPL
22460  * <script type="text/javascript">
22461  */
22462
22463
22464 /**
22465  * @class Roo.data.SortTypes
22466  * @singleton
22467  * Defines the default sorting (casting?) comparison functions used when sorting data.
22468  */
22469 Roo.data.SortTypes = {
22470     /**
22471      * Default sort that does nothing
22472      * @param {Mixed} s The value being converted
22473      * @return {Mixed} The comparison value
22474      */
22475     none : function(s){
22476         return s;
22477     },
22478     
22479     /**
22480      * The regular expression used to strip tags
22481      * @type {RegExp}
22482      * @property
22483      */
22484     stripTagsRE : /<\/?[^>]+>/gi,
22485     
22486     /**
22487      * Strips all HTML tags to sort on text only
22488      * @param {Mixed} s The value being converted
22489      * @return {String} The comparison value
22490      */
22491     asText : function(s){
22492         return String(s).replace(this.stripTagsRE, "");
22493     },
22494     
22495     /**
22496      * Strips all HTML tags to sort on text only - Case insensitive
22497      * @param {Mixed} s The value being converted
22498      * @return {String} The comparison value
22499      */
22500     asUCText : function(s){
22501         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22502     },
22503     
22504     /**
22505      * Case insensitive string
22506      * @param {Mixed} s The value being converted
22507      * @return {String} The comparison value
22508      */
22509     asUCString : function(s) {
22510         return String(s).toUpperCase();
22511     },
22512     
22513     /**
22514      * Date sorting
22515      * @param {Mixed} s The value being converted
22516      * @return {Number} The comparison value
22517      */
22518     asDate : function(s) {
22519         if(!s){
22520             return 0;
22521         }
22522         if(s instanceof Date){
22523             return s.getTime();
22524         }
22525         return Date.parse(String(s));
22526     },
22527     
22528     /**
22529      * Float sorting
22530      * @param {Mixed} s The value being converted
22531      * @return {Float} The comparison value
22532      */
22533     asFloat : function(s) {
22534         var val = parseFloat(String(s).replace(/,/g, ""));
22535         if(isNaN(val)) {
22536             val = 0;
22537         }
22538         return val;
22539     },
22540     
22541     /**
22542      * Integer sorting
22543      * @param {Mixed} s The value being converted
22544      * @return {Number} The comparison value
22545      */
22546     asInt : function(s) {
22547         var val = parseInt(String(s).replace(/,/g, ""));
22548         if(isNaN(val)) {
22549             val = 0;
22550         }
22551         return val;
22552     }
22553 };/*
22554  * Based on:
22555  * Ext JS Library 1.1.1
22556  * Copyright(c) 2006-2007, Ext JS, LLC.
22557  *
22558  * Originally Released Under LGPL - original licence link has changed is not relivant.
22559  *
22560  * Fork - LGPL
22561  * <script type="text/javascript">
22562  */
22563
22564 /**
22565 * @class Roo.data.Record
22566  * Instances of this class encapsulate both record <em>definition</em> information, and record
22567  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22568  * to access Records cached in an {@link Roo.data.Store} object.<br>
22569  * <p>
22570  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22571  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22572  * objects.<br>
22573  * <p>
22574  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22575  * @constructor
22576  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22577  * {@link #create}. The parameters are the same.
22578  * @param {Array} data An associative Array of data values keyed by the field name.
22579  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22580  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22581  * not specified an integer id is generated.
22582  */
22583 Roo.data.Record = function(data, id){
22584     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22585     this.data = data;
22586 };
22587
22588 /**
22589  * Generate a constructor for a specific record layout.
22590  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22591  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22592  * Each field definition object may contain the following properties: <ul>
22593  * <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,
22594  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22595  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22596  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22597  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22598  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22599  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22600  * this may be omitted.</p></li>
22601  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22602  * <ul><li>auto (Default, implies no conversion)</li>
22603  * <li>string</li>
22604  * <li>int</li>
22605  * <li>float</li>
22606  * <li>boolean</li>
22607  * <li>date</li></ul></p></li>
22608  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22609  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22610  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22611  * by the Reader into an object that will be stored in the Record. It is passed the
22612  * following parameters:<ul>
22613  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22614  * </ul></p></li>
22615  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22616  * </ul>
22617  * <br>usage:<br><pre><code>
22618 var TopicRecord = Roo.data.Record.create(
22619     {name: 'title', mapping: 'topic_title'},
22620     {name: 'author', mapping: 'username'},
22621     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22622     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22623     {name: 'lastPoster', mapping: 'user2'},
22624     {name: 'excerpt', mapping: 'post_text'}
22625 );
22626
22627 var myNewRecord = new TopicRecord({
22628     title: 'Do my job please',
22629     author: 'noobie',
22630     totalPosts: 1,
22631     lastPost: new Date(),
22632     lastPoster: 'Animal',
22633     excerpt: 'No way dude!'
22634 });
22635 myStore.add(myNewRecord);
22636 </code></pre>
22637  * @method create
22638  * @static
22639  */
22640 Roo.data.Record.create = function(o){
22641     var f = function(){
22642         f.superclass.constructor.apply(this, arguments);
22643     };
22644     Roo.extend(f, Roo.data.Record);
22645     var p = f.prototype;
22646     p.fields = new Roo.util.MixedCollection(false, function(field){
22647         return field.name;
22648     });
22649     for(var i = 0, len = o.length; i < len; i++){
22650         p.fields.add(new Roo.data.Field(o[i]));
22651     }
22652     f.getField = function(name){
22653         return p.fields.get(name);  
22654     };
22655     return f;
22656 };
22657
22658 Roo.data.Record.AUTO_ID = 1000;
22659 Roo.data.Record.EDIT = 'edit';
22660 Roo.data.Record.REJECT = 'reject';
22661 Roo.data.Record.COMMIT = 'commit';
22662
22663 Roo.data.Record.prototype = {
22664     /**
22665      * Readonly flag - true if this record has been modified.
22666      * @type Boolean
22667      */
22668     dirty : false,
22669     editing : false,
22670     error: null,
22671     modified: null,
22672
22673     // private
22674     join : function(store){
22675         this.store = store;
22676     },
22677
22678     /**
22679      * Set the named field to the specified value.
22680      * @param {String} name The name of the field to set.
22681      * @param {Object} value The value to set the field to.
22682      */
22683     set : function(name, value){
22684         if(this.data[name] == value){
22685             return;
22686         }
22687         this.dirty = true;
22688         if(!this.modified){
22689             this.modified = {};
22690         }
22691         if(typeof this.modified[name] == 'undefined'){
22692             this.modified[name] = this.data[name];
22693         }
22694         this.data[name] = value;
22695         if(!this.editing && this.store){
22696             this.store.afterEdit(this);
22697         }       
22698     },
22699
22700     /**
22701      * Get the value of the named field.
22702      * @param {String} name The name of the field to get the value of.
22703      * @return {Object} The value of the field.
22704      */
22705     get : function(name){
22706         return this.data[name]; 
22707     },
22708
22709     // private
22710     beginEdit : function(){
22711         this.editing = true;
22712         this.modified = {}; 
22713     },
22714
22715     // private
22716     cancelEdit : function(){
22717         this.editing = false;
22718         delete this.modified;
22719     },
22720
22721     // private
22722     endEdit : function(){
22723         this.editing = false;
22724         if(this.dirty && this.store){
22725             this.store.afterEdit(this);
22726         }
22727     },
22728
22729     /**
22730      * Usually called by the {@link Roo.data.Store} which owns the Record.
22731      * Rejects all changes made to the Record since either creation, or the last commit operation.
22732      * Modified fields are reverted to their original values.
22733      * <p>
22734      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22735      * of reject operations.
22736      */
22737     reject : function(){
22738         var m = this.modified;
22739         for(var n in m){
22740             if(typeof m[n] != "function"){
22741                 this.data[n] = m[n];
22742             }
22743         }
22744         this.dirty = false;
22745         delete this.modified;
22746         this.editing = false;
22747         if(this.store){
22748             this.store.afterReject(this);
22749         }
22750     },
22751
22752     /**
22753      * Usually called by the {@link Roo.data.Store} which owns the Record.
22754      * Commits all changes made to the Record since either creation, or the last commit operation.
22755      * <p>
22756      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22757      * of commit operations.
22758      */
22759     commit : function(){
22760         this.dirty = false;
22761         delete this.modified;
22762         this.editing = false;
22763         if(this.store){
22764             this.store.afterCommit(this);
22765         }
22766     },
22767
22768     // private
22769     hasError : function(){
22770         return this.error != null;
22771     },
22772
22773     // private
22774     clearError : function(){
22775         this.error = null;
22776     },
22777
22778     /**
22779      * Creates a copy of this record.
22780      * @param {String} id (optional) A new record id if you don't want to use this record's id
22781      * @return {Record}
22782      */
22783     copy : function(newId) {
22784         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22785     }
22786 };/*
22787  * Based on:
22788  * Ext JS Library 1.1.1
22789  * Copyright(c) 2006-2007, Ext JS, LLC.
22790  *
22791  * Originally Released Under LGPL - original licence link has changed is not relivant.
22792  *
22793  * Fork - LGPL
22794  * <script type="text/javascript">
22795  */
22796
22797
22798
22799 /**
22800  * @class Roo.data.Store
22801  * @extends Roo.util.Observable
22802  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22803  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22804  * <p>
22805  * 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
22806  * has no knowledge of the format of the data returned by the Proxy.<br>
22807  * <p>
22808  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22809  * instances from the data object. These records are cached and made available through accessor functions.
22810  * @constructor
22811  * Creates a new Store.
22812  * @param {Object} config A config object containing the objects needed for the Store to access data,
22813  * and read the data into Records.
22814  */
22815 Roo.data.Store = function(config){
22816     this.data = new Roo.util.MixedCollection(false);
22817     this.data.getKey = function(o){
22818         return o.id;
22819     };
22820     this.baseParams = {};
22821     // private
22822     this.paramNames = {
22823         "start" : "start",
22824         "limit" : "limit",
22825         "sort" : "sort",
22826         "dir" : "dir",
22827         "multisort" : "_multisort"
22828     };
22829
22830     if(config && config.data){
22831         this.inlineData = config.data;
22832         delete config.data;
22833     }
22834
22835     Roo.apply(this, config);
22836     
22837     if(this.reader){ // reader passed
22838         this.reader = Roo.factory(this.reader, Roo.data);
22839         this.reader.xmodule = this.xmodule || false;
22840         if(!this.recordType){
22841             this.recordType = this.reader.recordType;
22842         }
22843         if(this.reader.onMetaChange){
22844             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22845         }
22846     }
22847
22848     if(this.recordType){
22849         this.fields = this.recordType.prototype.fields;
22850     }
22851     this.modified = [];
22852
22853     this.addEvents({
22854         /**
22855          * @event datachanged
22856          * Fires when the data cache has changed, and a widget which is using this Store
22857          * as a Record cache should refresh its view.
22858          * @param {Store} this
22859          */
22860         datachanged : true,
22861         /**
22862          * @event metachange
22863          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22864          * @param {Store} this
22865          * @param {Object} meta The JSON metadata
22866          */
22867         metachange : true,
22868         /**
22869          * @event add
22870          * Fires when Records have been added to the Store
22871          * @param {Store} this
22872          * @param {Roo.data.Record[]} records The array of Records added
22873          * @param {Number} index The index at which the record(s) were added
22874          */
22875         add : true,
22876         /**
22877          * @event remove
22878          * Fires when a Record has been removed from the Store
22879          * @param {Store} this
22880          * @param {Roo.data.Record} record The Record that was removed
22881          * @param {Number} index The index at which the record was removed
22882          */
22883         remove : true,
22884         /**
22885          * @event update
22886          * Fires when a Record has been updated
22887          * @param {Store} this
22888          * @param {Roo.data.Record} record The Record that was updated
22889          * @param {String} operation The update operation being performed.  Value may be one of:
22890          * <pre><code>
22891  Roo.data.Record.EDIT
22892  Roo.data.Record.REJECT
22893  Roo.data.Record.COMMIT
22894          * </code></pre>
22895          */
22896         update : true,
22897         /**
22898          * @event clear
22899          * Fires when the data cache has been cleared.
22900          * @param {Store} this
22901          */
22902         clear : true,
22903         /**
22904          * @event beforeload
22905          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22906          * the load action will be canceled.
22907          * @param {Store} this
22908          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22909          */
22910         beforeload : true,
22911         /**
22912          * @event beforeloadadd
22913          * Fires after a new set of Records has been loaded.
22914          * @param {Store} this
22915          * @param {Roo.data.Record[]} records The Records that were loaded
22916          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22917          */
22918         beforeloadadd : true,
22919         /**
22920          * @event load
22921          * Fires after a new set of Records has been loaded, before they are added to the store.
22922          * @param {Store} this
22923          * @param {Roo.data.Record[]} records The Records that were loaded
22924          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22925          * @params {Object} return from reader
22926          */
22927         load : true,
22928         /**
22929          * @event loadexception
22930          * Fires if an exception occurs in the Proxy during loading.
22931          * Called with the signature of the Proxy's "loadexception" event.
22932          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22933          * 
22934          * @param {Proxy} 
22935          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22936          * @param {Object} load options 
22937          * @param {Object} jsonData from your request (normally this contains the Exception)
22938          */
22939         loadexception : true
22940     });
22941     
22942     if(this.proxy){
22943         this.proxy = Roo.factory(this.proxy, Roo.data);
22944         this.proxy.xmodule = this.xmodule || false;
22945         this.relayEvents(this.proxy,  ["loadexception"]);
22946     }
22947     this.sortToggle = {};
22948     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22949
22950     Roo.data.Store.superclass.constructor.call(this);
22951
22952     if(this.inlineData){
22953         this.loadData(this.inlineData);
22954         delete this.inlineData;
22955     }
22956 };
22957
22958 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22959      /**
22960     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22961     * without a remote query - used by combo/forms at present.
22962     */
22963     
22964     /**
22965     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22966     */
22967     /**
22968     * @cfg {Array} data Inline data to be loaded when the store is initialized.
22969     */
22970     /**
22971     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
22972     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
22973     */
22974     /**
22975     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
22976     * on any HTTP request
22977     */
22978     /**
22979     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
22980     */
22981     /**
22982     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
22983     */
22984     multiSort: false,
22985     /**
22986     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
22987     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
22988     */
22989     remoteSort : false,
22990
22991     /**
22992     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
22993      * loaded or when a record is removed. (defaults to false).
22994     */
22995     pruneModifiedRecords : false,
22996
22997     // private
22998     lastOptions : null,
22999
23000     /**
23001      * Add Records to the Store and fires the add event.
23002      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23003      */
23004     add : function(records){
23005         records = [].concat(records);
23006         for(var i = 0, len = records.length; i < len; i++){
23007             records[i].join(this);
23008         }
23009         var index = this.data.length;
23010         this.data.addAll(records);
23011         this.fireEvent("add", this, records, index);
23012     },
23013
23014     /**
23015      * Remove a Record from the Store and fires the remove event.
23016      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23017      */
23018     remove : function(record){
23019         var index = this.data.indexOf(record);
23020         this.data.removeAt(index);
23021  
23022         if(this.pruneModifiedRecords){
23023             this.modified.remove(record);
23024         }
23025         this.fireEvent("remove", this, record, index);
23026     },
23027
23028     /**
23029      * Remove all Records from the Store and fires the clear event.
23030      */
23031     removeAll : function(){
23032         this.data.clear();
23033         if(this.pruneModifiedRecords){
23034             this.modified = [];
23035         }
23036         this.fireEvent("clear", this);
23037     },
23038
23039     /**
23040      * Inserts Records to the Store at the given index and fires the add event.
23041      * @param {Number} index The start index at which to insert the passed Records.
23042      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23043      */
23044     insert : function(index, records){
23045         records = [].concat(records);
23046         for(var i = 0, len = records.length; i < len; i++){
23047             this.data.insert(index, records[i]);
23048             records[i].join(this);
23049         }
23050         this.fireEvent("add", this, records, index);
23051     },
23052
23053     /**
23054      * Get the index within the cache of the passed Record.
23055      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23056      * @return {Number} The index of the passed Record. Returns -1 if not found.
23057      */
23058     indexOf : function(record){
23059         return this.data.indexOf(record);
23060     },
23061
23062     /**
23063      * Get the index within the cache of the Record with the passed id.
23064      * @param {String} id The id of the Record to find.
23065      * @return {Number} The index of the Record. Returns -1 if not found.
23066      */
23067     indexOfId : function(id){
23068         return this.data.indexOfKey(id);
23069     },
23070
23071     /**
23072      * Get the Record with the specified id.
23073      * @param {String} id The id of the Record to find.
23074      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23075      */
23076     getById : function(id){
23077         return this.data.key(id);
23078     },
23079
23080     /**
23081      * Get the Record at the specified index.
23082      * @param {Number} index The index of the Record to find.
23083      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23084      */
23085     getAt : function(index){
23086         return this.data.itemAt(index);
23087     },
23088
23089     /**
23090      * Returns a range of Records between specified indices.
23091      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23092      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23093      * @return {Roo.data.Record[]} An array of Records
23094      */
23095     getRange : function(start, end){
23096         return this.data.getRange(start, end);
23097     },
23098
23099     // private
23100     storeOptions : function(o){
23101         o = Roo.apply({}, o);
23102         delete o.callback;
23103         delete o.scope;
23104         this.lastOptions = o;
23105     },
23106
23107     /**
23108      * Loads the Record cache from the configured Proxy using the configured Reader.
23109      * <p>
23110      * If using remote paging, then the first load call must specify the <em>start</em>
23111      * and <em>limit</em> properties in the options.params property to establish the initial
23112      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23113      * <p>
23114      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23115      * and this call will return before the new data has been loaded. Perform any post-processing
23116      * in a callback function, or in a "load" event handler.</strong>
23117      * <p>
23118      * @param {Object} options An object containing properties which control loading options:<ul>
23119      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23120      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23121      * passed the following arguments:<ul>
23122      * <li>r : Roo.data.Record[]</li>
23123      * <li>options: Options object from the load call</li>
23124      * <li>success: Boolean success indicator</li></ul></li>
23125      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23126      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23127      * </ul>
23128      */
23129     load : function(options){
23130         options = options || {};
23131         if(this.fireEvent("beforeload", this, options) !== false){
23132             this.storeOptions(options);
23133             var p = Roo.apply(options.params || {}, this.baseParams);
23134             // if meta was not loaded from remote source.. try requesting it.
23135             if (!this.reader.metaFromRemote) {
23136                 p._requestMeta = 1;
23137             }
23138             if(this.sortInfo && this.remoteSort){
23139                 var pn = this.paramNames;
23140                 p[pn["sort"]] = this.sortInfo.field;
23141                 p[pn["dir"]] = this.sortInfo.direction;
23142             }
23143             if (this.multiSort) {
23144                 var pn = this.paramNames;
23145                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23146             }
23147             
23148             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23149         }
23150     },
23151
23152     /**
23153      * Reloads the Record cache from the configured Proxy using the configured Reader and
23154      * the options from the last load operation performed.
23155      * @param {Object} options (optional) An object containing properties which may override the options
23156      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23157      * the most recently used options are reused).
23158      */
23159     reload : function(options){
23160         this.load(Roo.applyIf(options||{}, this.lastOptions));
23161     },
23162
23163     // private
23164     // Called as a callback by the Reader during a load operation.
23165     loadRecords : function(o, options, success){
23166         if(!o || success === false){
23167             if(success !== false){
23168                 this.fireEvent("load", this, [], options, o);
23169             }
23170             if(options.callback){
23171                 options.callback.call(options.scope || this, [], options, false);
23172             }
23173             return;
23174         }
23175         // if data returned failure - throw an exception.
23176         if (o.success === false) {
23177             // show a message if no listener is registered.
23178             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23179                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23180             }
23181             // loadmask wil be hooked into this..
23182             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23183             return;
23184         }
23185         var r = o.records, t = o.totalRecords || r.length;
23186         
23187         this.fireEvent("beforeloadadd", this, r, options, o);
23188         
23189         if(!options || options.add !== true){
23190             if(this.pruneModifiedRecords){
23191                 this.modified = [];
23192             }
23193             for(var i = 0, len = r.length; i < len; i++){
23194                 r[i].join(this);
23195             }
23196             if(this.snapshot){
23197                 this.data = this.snapshot;
23198                 delete this.snapshot;
23199             }
23200             this.data.clear();
23201             this.data.addAll(r);
23202             this.totalLength = t;
23203             this.applySort();
23204             this.fireEvent("datachanged", this);
23205         }else{
23206             this.totalLength = Math.max(t, this.data.length+r.length);
23207             this.add(r);
23208         }
23209         
23210         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23211                 
23212             var e = new Roo.data.Record({});
23213
23214             e.set(this.parent.displayField, this.parent.emptyTitle);
23215             e.set(this.parent.valueField, '');
23216
23217             this.insert(0, e);
23218         }
23219             
23220         this.fireEvent("load", this, r, options, o);
23221         if(options.callback){
23222             options.callback.call(options.scope || this, r, options, true);
23223         }
23224     },
23225
23226
23227     /**
23228      * Loads data from a passed data block. A Reader which understands the format of the data
23229      * must have been configured in the constructor.
23230      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23231      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23232      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23233      */
23234     loadData : function(o, append){
23235         var r = this.reader.readRecords(o);
23236         this.loadRecords(r, {add: append}, true);
23237     },
23238
23239     /**
23240      * Gets the number of cached records.
23241      * <p>
23242      * <em>If using paging, this may not be the total size of the dataset. If the data object
23243      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23244      * the data set size</em>
23245      */
23246     getCount : function(){
23247         return this.data.length || 0;
23248     },
23249
23250     /**
23251      * Gets the total number of records in the dataset as returned by the server.
23252      * <p>
23253      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23254      * the dataset size</em>
23255      */
23256     getTotalCount : function(){
23257         return this.totalLength || 0;
23258     },
23259
23260     /**
23261      * Returns the sort state of the Store as an object with two properties:
23262      * <pre><code>
23263  field {String} The name of the field by which the Records are sorted
23264  direction {String} The sort order, "ASC" or "DESC"
23265      * </code></pre>
23266      */
23267     getSortState : function(){
23268         return this.sortInfo;
23269     },
23270
23271     // private
23272     applySort : function(){
23273         if(this.sortInfo && !this.remoteSort){
23274             var s = this.sortInfo, f = s.field;
23275             var st = this.fields.get(f).sortType;
23276             var fn = function(r1, r2){
23277                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23278                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23279             };
23280             this.data.sort(s.direction, fn);
23281             if(this.snapshot && this.snapshot != this.data){
23282                 this.snapshot.sort(s.direction, fn);
23283             }
23284         }
23285     },
23286
23287     /**
23288      * Sets the default sort column and order to be used by the next load operation.
23289      * @param {String} fieldName The name of the field to sort by.
23290      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23291      */
23292     setDefaultSort : function(field, dir){
23293         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23294     },
23295
23296     /**
23297      * Sort the Records.
23298      * If remote sorting is used, the sort is performed on the server, and the cache is
23299      * reloaded. If local sorting is used, the cache is sorted internally.
23300      * @param {String} fieldName The name of the field to sort by.
23301      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23302      */
23303     sort : function(fieldName, dir){
23304         var f = this.fields.get(fieldName);
23305         if(!dir){
23306             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23307             
23308             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23309                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23310             }else{
23311                 dir = f.sortDir;
23312             }
23313         }
23314         this.sortToggle[f.name] = dir;
23315         this.sortInfo = {field: f.name, direction: dir};
23316         if(!this.remoteSort){
23317             this.applySort();
23318             this.fireEvent("datachanged", this);
23319         }else{
23320             this.load(this.lastOptions);
23321         }
23322     },
23323
23324     /**
23325      * Calls the specified function for each of the Records in the cache.
23326      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23327      * Returning <em>false</em> aborts and exits the iteration.
23328      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23329      */
23330     each : function(fn, scope){
23331         this.data.each(fn, scope);
23332     },
23333
23334     /**
23335      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23336      * (e.g., during paging).
23337      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23338      */
23339     getModifiedRecords : function(){
23340         return this.modified;
23341     },
23342
23343     // private
23344     createFilterFn : function(property, value, anyMatch){
23345         if(!value.exec){ // not a regex
23346             value = String(value);
23347             if(value.length == 0){
23348                 return false;
23349             }
23350             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23351         }
23352         return function(r){
23353             return value.test(r.data[property]);
23354         };
23355     },
23356
23357     /**
23358      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23359      * @param {String} property A field on your records
23360      * @param {Number} start The record index to start at (defaults to 0)
23361      * @param {Number} end The last record index to include (defaults to length - 1)
23362      * @return {Number} The sum
23363      */
23364     sum : function(property, start, end){
23365         var rs = this.data.items, v = 0;
23366         start = start || 0;
23367         end = (end || end === 0) ? end : rs.length-1;
23368
23369         for(var i = start; i <= end; i++){
23370             v += (rs[i].data[property] || 0);
23371         }
23372         return v;
23373     },
23374
23375     /**
23376      * Filter the records by a specified property.
23377      * @param {String} field A field on your records
23378      * @param {String/RegExp} value Either a string that the field
23379      * should start with or a RegExp to test against the field
23380      * @param {Boolean} anyMatch True to match any part not just the beginning
23381      */
23382     filter : function(property, value, anyMatch){
23383         var fn = this.createFilterFn(property, value, anyMatch);
23384         return fn ? this.filterBy(fn) : this.clearFilter();
23385     },
23386
23387     /**
23388      * Filter by a function. The specified function will be called with each
23389      * record in this data source. If the function returns true the record is included,
23390      * otherwise it is filtered.
23391      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23392      * @param {Object} scope (optional) The scope of the function (defaults to this)
23393      */
23394     filterBy : function(fn, scope){
23395         this.snapshot = this.snapshot || this.data;
23396         this.data = this.queryBy(fn, scope||this);
23397         this.fireEvent("datachanged", this);
23398     },
23399
23400     /**
23401      * Query the records by a specified property.
23402      * @param {String} field A field on your records
23403      * @param {String/RegExp} value Either a string that the field
23404      * should start with or a RegExp to test against the field
23405      * @param {Boolean} anyMatch True to match any part not just the beginning
23406      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23407      */
23408     query : function(property, value, anyMatch){
23409         var fn = this.createFilterFn(property, value, anyMatch);
23410         return fn ? this.queryBy(fn) : this.data.clone();
23411     },
23412
23413     /**
23414      * Query by a function. The specified function will be called with each
23415      * record in this data source. If the function returns true the record is included
23416      * in the results.
23417      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23418      * @param {Object} scope (optional) The scope of the function (defaults to this)
23419       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23420      **/
23421     queryBy : function(fn, scope){
23422         var data = this.snapshot || this.data;
23423         return data.filterBy(fn, scope||this);
23424     },
23425
23426     /**
23427      * Collects unique values for a particular dataIndex from this store.
23428      * @param {String} dataIndex The property to collect
23429      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23430      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23431      * @return {Array} An array of the unique values
23432      **/
23433     collect : function(dataIndex, allowNull, bypassFilter){
23434         var d = (bypassFilter === true && this.snapshot) ?
23435                 this.snapshot.items : this.data.items;
23436         var v, sv, r = [], l = {};
23437         for(var i = 0, len = d.length; i < len; i++){
23438             v = d[i].data[dataIndex];
23439             sv = String(v);
23440             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23441                 l[sv] = true;
23442                 r[r.length] = v;
23443             }
23444         }
23445         return r;
23446     },
23447
23448     /**
23449      * Revert to a view of the Record cache with no filtering applied.
23450      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23451      */
23452     clearFilter : function(suppressEvent){
23453         if(this.snapshot && this.snapshot != this.data){
23454             this.data = this.snapshot;
23455             delete this.snapshot;
23456             if(suppressEvent !== true){
23457                 this.fireEvent("datachanged", this);
23458             }
23459         }
23460     },
23461
23462     // private
23463     afterEdit : function(record){
23464         if(this.modified.indexOf(record) == -1){
23465             this.modified.push(record);
23466         }
23467         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23468     },
23469     
23470     // private
23471     afterReject : function(record){
23472         this.modified.remove(record);
23473         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23474     },
23475
23476     // private
23477     afterCommit : function(record){
23478         this.modified.remove(record);
23479         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23480     },
23481
23482     /**
23483      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23484      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23485      */
23486     commitChanges : function(){
23487         var m = this.modified.slice(0);
23488         this.modified = [];
23489         for(var i = 0, len = m.length; i < len; i++){
23490             m[i].commit();
23491         }
23492     },
23493
23494     /**
23495      * Cancel outstanding changes on all changed records.
23496      */
23497     rejectChanges : function(){
23498         var m = this.modified.slice(0);
23499         this.modified = [];
23500         for(var i = 0, len = m.length; i < len; i++){
23501             m[i].reject();
23502         }
23503     },
23504
23505     onMetaChange : function(meta, rtype, o){
23506         this.recordType = rtype;
23507         this.fields = rtype.prototype.fields;
23508         delete this.snapshot;
23509         this.sortInfo = meta.sortInfo || this.sortInfo;
23510         this.modified = [];
23511         this.fireEvent('metachange', this, this.reader.meta);
23512     },
23513     
23514     moveIndex : function(data, type)
23515     {
23516         var index = this.indexOf(data);
23517         
23518         var newIndex = index + type;
23519         
23520         this.remove(data);
23521         
23522         this.insert(newIndex, data);
23523         
23524     }
23525 });/*
23526  * Based on:
23527  * Ext JS Library 1.1.1
23528  * Copyright(c) 2006-2007, Ext JS, LLC.
23529  *
23530  * Originally Released Under LGPL - original licence link has changed is not relivant.
23531  *
23532  * Fork - LGPL
23533  * <script type="text/javascript">
23534  */
23535
23536 /**
23537  * @class Roo.data.SimpleStore
23538  * @extends Roo.data.Store
23539  * Small helper class to make creating Stores from Array data easier.
23540  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23541  * @cfg {Array} fields An array of field definition objects, or field name strings.
23542  * @cfg {Array} data The multi-dimensional array of data
23543  * @constructor
23544  * @param {Object} config
23545  */
23546 Roo.data.SimpleStore = function(config){
23547     Roo.data.SimpleStore.superclass.constructor.call(this, {
23548         isLocal : true,
23549         reader: new Roo.data.ArrayReader({
23550                 id: config.id
23551             },
23552             Roo.data.Record.create(config.fields)
23553         ),
23554         proxy : new Roo.data.MemoryProxy(config.data)
23555     });
23556     this.load();
23557 };
23558 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23559  * Based on:
23560  * Ext JS Library 1.1.1
23561  * Copyright(c) 2006-2007, Ext JS, LLC.
23562  *
23563  * Originally Released Under LGPL - original licence link has changed is not relivant.
23564  *
23565  * Fork - LGPL
23566  * <script type="text/javascript">
23567  */
23568
23569 /**
23570 /**
23571  * @extends Roo.data.Store
23572  * @class Roo.data.JsonStore
23573  * Small helper class to make creating Stores for JSON data easier. <br/>
23574 <pre><code>
23575 var store = new Roo.data.JsonStore({
23576     url: 'get-images.php',
23577     root: 'images',
23578     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23579 });
23580 </code></pre>
23581  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23582  * JsonReader and HttpProxy (unless inline data is provided).</b>
23583  * @cfg {Array} fields An array of field definition objects, or field name strings.
23584  * @constructor
23585  * @param {Object} config
23586  */
23587 Roo.data.JsonStore = function(c){
23588     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23589         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23590         reader: new Roo.data.JsonReader(c, c.fields)
23591     }));
23592 };
23593 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23594  * Based on:
23595  * Ext JS Library 1.1.1
23596  * Copyright(c) 2006-2007, Ext JS, LLC.
23597  *
23598  * Originally Released Under LGPL - original licence link has changed is not relivant.
23599  *
23600  * Fork - LGPL
23601  * <script type="text/javascript">
23602  */
23603
23604  
23605 Roo.data.Field = function(config){
23606     if(typeof config == "string"){
23607         config = {name: config};
23608     }
23609     Roo.apply(this, config);
23610     
23611     if(!this.type){
23612         this.type = "auto";
23613     }
23614     
23615     var st = Roo.data.SortTypes;
23616     // named sortTypes are supported, here we look them up
23617     if(typeof this.sortType == "string"){
23618         this.sortType = st[this.sortType];
23619     }
23620     
23621     // set default sortType for strings and dates
23622     if(!this.sortType){
23623         switch(this.type){
23624             case "string":
23625                 this.sortType = st.asUCString;
23626                 break;
23627             case "date":
23628                 this.sortType = st.asDate;
23629                 break;
23630             default:
23631                 this.sortType = st.none;
23632         }
23633     }
23634
23635     // define once
23636     var stripRe = /[\$,%]/g;
23637
23638     // prebuilt conversion function for this field, instead of
23639     // switching every time we're reading a value
23640     if(!this.convert){
23641         var cv, dateFormat = this.dateFormat;
23642         switch(this.type){
23643             case "":
23644             case "auto":
23645             case undefined:
23646                 cv = function(v){ return v; };
23647                 break;
23648             case "string":
23649                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23650                 break;
23651             case "int":
23652                 cv = function(v){
23653                     return v !== undefined && v !== null && v !== '' ?
23654                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23655                     };
23656                 break;
23657             case "float":
23658                 cv = function(v){
23659                     return v !== undefined && v !== null && v !== '' ?
23660                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23661                     };
23662                 break;
23663             case "bool":
23664             case "boolean":
23665                 cv = function(v){ return v === true || v === "true" || v == 1; };
23666                 break;
23667             case "date":
23668                 cv = function(v){
23669                     if(!v){
23670                         return '';
23671                     }
23672                     if(v instanceof Date){
23673                         return v;
23674                     }
23675                     if(dateFormat){
23676                         if(dateFormat == "timestamp"){
23677                             return new Date(v*1000);
23678                         }
23679                         return Date.parseDate(v, dateFormat);
23680                     }
23681                     var parsed = Date.parse(v);
23682                     return parsed ? new Date(parsed) : null;
23683                 };
23684              break;
23685             
23686         }
23687         this.convert = cv;
23688     }
23689 };
23690
23691 Roo.data.Field.prototype = {
23692     dateFormat: null,
23693     defaultValue: "",
23694     mapping: null,
23695     sortType : null,
23696     sortDir : "ASC"
23697 };/*
23698  * Based on:
23699  * Ext JS Library 1.1.1
23700  * Copyright(c) 2006-2007, Ext JS, LLC.
23701  *
23702  * Originally Released Under LGPL - original licence link has changed is not relivant.
23703  *
23704  * Fork - LGPL
23705  * <script type="text/javascript">
23706  */
23707  
23708 // Base class for reading structured data from a data source.  This class is intended to be
23709 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23710
23711 /**
23712  * @class Roo.data.DataReader
23713  * Base class for reading structured data from a data source.  This class is intended to be
23714  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23715  */
23716
23717 Roo.data.DataReader = function(meta, recordType){
23718     
23719     this.meta = meta;
23720     
23721     this.recordType = recordType instanceof Array ? 
23722         Roo.data.Record.create(recordType) : recordType;
23723 };
23724
23725 Roo.data.DataReader.prototype = {
23726      /**
23727      * Create an empty record
23728      * @param {Object} data (optional) - overlay some values
23729      * @return {Roo.data.Record} record created.
23730      */
23731     newRow :  function(d) {
23732         var da =  {};
23733         this.recordType.prototype.fields.each(function(c) {
23734             switch( c.type) {
23735                 case 'int' : da[c.name] = 0; break;
23736                 case 'date' : da[c.name] = new Date(); break;
23737                 case 'float' : da[c.name] = 0.0; break;
23738                 case 'boolean' : da[c.name] = false; break;
23739                 default : da[c.name] = ""; break;
23740             }
23741             
23742         });
23743         return new this.recordType(Roo.apply(da, d));
23744     }
23745     
23746 };/*
23747  * Based on:
23748  * Ext JS Library 1.1.1
23749  * Copyright(c) 2006-2007, Ext JS, LLC.
23750  *
23751  * Originally Released Under LGPL - original licence link has changed is not relivant.
23752  *
23753  * Fork - LGPL
23754  * <script type="text/javascript">
23755  */
23756
23757 /**
23758  * @class Roo.data.DataProxy
23759  * @extends Roo.data.Observable
23760  * This class is an abstract base class for implementations which provide retrieval of
23761  * unformatted data objects.<br>
23762  * <p>
23763  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23764  * (of the appropriate type which knows how to parse the data object) to provide a block of
23765  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23766  * <p>
23767  * Custom implementations must implement the load method as described in
23768  * {@link Roo.data.HttpProxy#load}.
23769  */
23770 Roo.data.DataProxy = function(){
23771     this.addEvents({
23772         /**
23773          * @event beforeload
23774          * Fires before a network request is made to retrieve a data object.
23775          * @param {Object} This DataProxy object.
23776          * @param {Object} params The params parameter to the load function.
23777          */
23778         beforeload : true,
23779         /**
23780          * @event load
23781          * Fires before the load method's callback is called.
23782          * @param {Object} This DataProxy object.
23783          * @param {Object} o The data object.
23784          * @param {Object} arg The callback argument object passed to the load function.
23785          */
23786         load : true,
23787         /**
23788          * @event loadexception
23789          * Fires if an Exception occurs during data retrieval.
23790          * @param {Object} This DataProxy object.
23791          * @param {Object} o The data object.
23792          * @param {Object} arg The callback argument object passed to the load function.
23793          * @param {Object} e The Exception.
23794          */
23795         loadexception : true
23796     });
23797     Roo.data.DataProxy.superclass.constructor.call(this);
23798 };
23799
23800 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23801
23802     /**
23803      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23804      */
23805 /*
23806  * Based on:
23807  * Ext JS Library 1.1.1
23808  * Copyright(c) 2006-2007, Ext JS, LLC.
23809  *
23810  * Originally Released Under LGPL - original licence link has changed is not relivant.
23811  *
23812  * Fork - LGPL
23813  * <script type="text/javascript">
23814  */
23815 /**
23816  * @class Roo.data.MemoryProxy
23817  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23818  * to the Reader when its load method is called.
23819  * @constructor
23820  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23821  */
23822 Roo.data.MemoryProxy = function(data){
23823     if (data.data) {
23824         data = data.data;
23825     }
23826     Roo.data.MemoryProxy.superclass.constructor.call(this);
23827     this.data = data;
23828 };
23829
23830 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23831     
23832     /**
23833      * Load data from the requested source (in this case an in-memory
23834      * data object passed to the constructor), read the data object into
23835      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23836      * process that block using the passed callback.
23837      * @param {Object} params This parameter is not used by the MemoryProxy class.
23838      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23839      * object into a block of Roo.data.Records.
23840      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23841      * The function must be passed <ul>
23842      * <li>The Record block object</li>
23843      * <li>The "arg" argument from the load function</li>
23844      * <li>A boolean success indicator</li>
23845      * </ul>
23846      * @param {Object} scope The scope in which to call the callback
23847      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23848      */
23849     load : function(params, reader, callback, scope, arg){
23850         params = params || {};
23851         var result;
23852         try {
23853             result = reader.readRecords(params.data ? params.data :this.data);
23854         }catch(e){
23855             this.fireEvent("loadexception", this, arg, null, e);
23856             callback.call(scope, null, arg, false);
23857             return;
23858         }
23859         callback.call(scope, result, arg, true);
23860     },
23861     
23862     // private
23863     update : function(params, records){
23864         
23865     }
23866 });/*
23867  * Based on:
23868  * Ext JS Library 1.1.1
23869  * Copyright(c) 2006-2007, Ext JS, LLC.
23870  *
23871  * Originally Released Under LGPL - original licence link has changed is not relivant.
23872  *
23873  * Fork - LGPL
23874  * <script type="text/javascript">
23875  */
23876 /**
23877  * @class Roo.data.HttpProxy
23878  * @extends Roo.data.DataProxy
23879  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23880  * configured to reference a certain URL.<br><br>
23881  * <p>
23882  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23883  * from which the running page was served.<br><br>
23884  * <p>
23885  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23886  * <p>
23887  * Be aware that to enable the browser to parse an XML document, the server must set
23888  * the Content-Type header in the HTTP response to "text/xml".
23889  * @constructor
23890  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23891  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23892  * will be used to make the request.
23893  */
23894 Roo.data.HttpProxy = function(conn){
23895     Roo.data.HttpProxy.superclass.constructor.call(this);
23896     // is conn a conn config or a real conn?
23897     this.conn = conn;
23898     this.useAjax = !conn || !conn.events;
23899   
23900 };
23901
23902 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23903     // thse are take from connection...
23904     
23905     /**
23906      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23907      */
23908     /**
23909      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23910      * extra parameters to each request made by this object. (defaults to undefined)
23911      */
23912     /**
23913      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23914      *  to each request made by this object. (defaults to undefined)
23915      */
23916     /**
23917      * @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)
23918      */
23919     /**
23920      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23921      */
23922      /**
23923      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23924      * @type Boolean
23925      */
23926   
23927
23928     /**
23929      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23930      * @type Boolean
23931      */
23932     /**
23933      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23934      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23935      * a finer-grained basis than the DataProxy events.
23936      */
23937     getConnection : function(){
23938         return this.useAjax ? Roo.Ajax : this.conn;
23939     },
23940
23941     /**
23942      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23943      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23944      * process that block using the passed callback.
23945      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23946      * for the request to the remote server.
23947      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23948      * object into a block of Roo.data.Records.
23949      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23950      * The function must be passed <ul>
23951      * <li>The Record block object</li>
23952      * <li>The "arg" argument from the load function</li>
23953      * <li>A boolean success indicator</li>
23954      * </ul>
23955      * @param {Object} scope The scope in which to call the callback
23956      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23957      */
23958     load : function(params, reader, callback, scope, arg){
23959         if(this.fireEvent("beforeload", this, params) !== false){
23960             var  o = {
23961                 params : params || {},
23962                 request: {
23963                     callback : callback,
23964                     scope : scope,
23965                     arg : arg
23966                 },
23967                 reader: reader,
23968                 callback : this.loadResponse,
23969                 scope: this
23970             };
23971             if(this.useAjax){
23972                 Roo.applyIf(o, this.conn);
23973                 if(this.activeRequest){
23974                     Roo.Ajax.abort(this.activeRequest);
23975                 }
23976                 this.activeRequest = Roo.Ajax.request(o);
23977             }else{
23978                 this.conn.request(o);
23979             }
23980         }else{
23981             callback.call(scope||this, null, arg, false);
23982         }
23983     },
23984
23985     // private
23986     loadResponse : function(o, success, response){
23987         delete this.activeRequest;
23988         if(!success){
23989             this.fireEvent("loadexception", this, o, response);
23990             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23991             return;
23992         }
23993         var result;
23994         try {
23995             result = o.reader.read(response);
23996         }catch(e){
23997             this.fireEvent("loadexception", this, o, response, e);
23998             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23999             return;
24000         }
24001         
24002         this.fireEvent("load", this, o, o.request.arg);
24003         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24004     },
24005
24006     // private
24007     update : function(dataSet){
24008
24009     },
24010
24011     // private
24012     updateResponse : function(dataSet){
24013
24014     }
24015 });/*
24016  * Based on:
24017  * Ext JS Library 1.1.1
24018  * Copyright(c) 2006-2007, Ext JS, LLC.
24019  *
24020  * Originally Released Under LGPL - original licence link has changed is not relivant.
24021  *
24022  * Fork - LGPL
24023  * <script type="text/javascript">
24024  */
24025
24026 /**
24027  * @class Roo.data.ScriptTagProxy
24028  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24029  * other than the originating domain of the running page.<br><br>
24030  * <p>
24031  * <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
24032  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24033  * <p>
24034  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24035  * source code that is used as the source inside a &lt;script> tag.<br><br>
24036  * <p>
24037  * In order for the browser to process the returned data, the server must wrap the data object
24038  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24039  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24040  * depending on whether the callback name was passed:
24041  * <p>
24042  * <pre><code>
24043 boolean scriptTag = false;
24044 String cb = request.getParameter("callback");
24045 if (cb != null) {
24046     scriptTag = true;
24047     response.setContentType("text/javascript");
24048 } else {
24049     response.setContentType("application/x-json");
24050 }
24051 Writer out = response.getWriter();
24052 if (scriptTag) {
24053     out.write(cb + "(");
24054 }
24055 out.print(dataBlock.toJsonString());
24056 if (scriptTag) {
24057     out.write(");");
24058 }
24059 </pre></code>
24060  *
24061  * @constructor
24062  * @param {Object} config A configuration object.
24063  */
24064 Roo.data.ScriptTagProxy = function(config){
24065     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24066     Roo.apply(this, config);
24067     this.head = document.getElementsByTagName("head")[0];
24068 };
24069
24070 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24071
24072 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24073     /**
24074      * @cfg {String} url The URL from which to request the data object.
24075      */
24076     /**
24077      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24078      */
24079     timeout : 30000,
24080     /**
24081      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24082      * the server the name of the callback function set up by the load call to process the returned data object.
24083      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24084      * javascript output which calls this named function passing the data object as its only parameter.
24085      */
24086     callbackParam : "callback",
24087     /**
24088      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24089      * name to the request.
24090      */
24091     nocache : true,
24092
24093     /**
24094      * Load data from the configured URL, read the data object into
24095      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24096      * process that block using the passed callback.
24097      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24098      * for the request to the remote server.
24099      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24100      * object into a block of Roo.data.Records.
24101      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24102      * The function must be passed <ul>
24103      * <li>The Record block object</li>
24104      * <li>The "arg" argument from the load function</li>
24105      * <li>A boolean success indicator</li>
24106      * </ul>
24107      * @param {Object} scope The scope in which to call the callback
24108      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24109      */
24110     load : function(params, reader, callback, scope, arg){
24111         if(this.fireEvent("beforeload", this, params) !== false){
24112
24113             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24114
24115             var url = this.url;
24116             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24117             if(this.nocache){
24118                 url += "&_dc=" + (new Date().getTime());
24119             }
24120             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24121             var trans = {
24122                 id : transId,
24123                 cb : "stcCallback"+transId,
24124                 scriptId : "stcScript"+transId,
24125                 params : params,
24126                 arg : arg,
24127                 url : url,
24128                 callback : callback,
24129                 scope : scope,
24130                 reader : reader
24131             };
24132             var conn = this;
24133
24134             window[trans.cb] = function(o){
24135                 conn.handleResponse(o, trans);
24136             };
24137
24138             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24139
24140             if(this.autoAbort !== false){
24141                 this.abort();
24142             }
24143
24144             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24145
24146             var script = document.createElement("script");
24147             script.setAttribute("src", url);
24148             script.setAttribute("type", "text/javascript");
24149             script.setAttribute("id", trans.scriptId);
24150             this.head.appendChild(script);
24151
24152             this.trans = trans;
24153         }else{
24154             callback.call(scope||this, null, arg, false);
24155         }
24156     },
24157
24158     // private
24159     isLoading : function(){
24160         return this.trans ? true : false;
24161     },
24162
24163     /**
24164      * Abort the current server request.
24165      */
24166     abort : function(){
24167         if(this.isLoading()){
24168             this.destroyTrans(this.trans);
24169         }
24170     },
24171
24172     // private
24173     destroyTrans : function(trans, isLoaded){
24174         this.head.removeChild(document.getElementById(trans.scriptId));
24175         clearTimeout(trans.timeoutId);
24176         if(isLoaded){
24177             window[trans.cb] = undefined;
24178             try{
24179                 delete window[trans.cb];
24180             }catch(e){}
24181         }else{
24182             // if hasn't been loaded, wait for load to remove it to prevent script error
24183             window[trans.cb] = function(){
24184                 window[trans.cb] = undefined;
24185                 try{
24186                     delete window[trans.cb];
24187                 }catch(e){}
24188             };
24189         }
24190     },
24191
24192     // private
24193     handleResponse : function(o, trans){
24194         this.trans = false;
24195         this.destroyTrans(trans, true);
24196         var result;
24197         try {
24198             result = trans.reader.readRecords(o);
24199         }catch(e){
24200             this.fireEvent("loadexception", this, o, trans.arg, e);
24201             trans.callback.call(trans.scope||window, null, trans.arg, false);
24202             return;
24203         }
24204         this.fireEvent("load", this, o, trans.arg);
24205         trans.callback.call(trans.scope||window, result, trans.arg, true);
24206     },
24207
24208     // private
24209     handleFailure : function(trans){
24210         this.trans = false;
24211         this.destroyTrans(trans, false);
24212         this.fireEvent("loadexception", this, null, trans.arg);
24213         trans.callback.call(trans.scope||window, null, trans.arg, false);
24214     }
24215 });/*
24216  * Based on:
24217  * Ext JS Library 1.1.1
24218  * Copyright(c) 2006-2007, Ext JS, LLC.
24219  *
24220  * Originally Released Under LGPL - original licence link has changed is not relivant.
24221  *
24222  * Fork - LGPL
24223  * <script type="text/javascript">
24224  */
24225
24226 /**
24227  * @class Roo.data.JsonReader
24228  * @extends Roo.data.DataReader
24229  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24230  * based on mappings in a provided Roo.data.Record constructor.
24231  * 
24232  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24233  * in the reply previously. 
24234  * 
24235  * <p>
24236  * Example code:
24237  * <pre><code>
24238 var RecordDef = Roo.data.Record.create([
24239     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24240     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24241 ]);
24242 var myReader = new Roo.data.JsonReader({
24243     totalProperty: "results",    // The property which contains the total dataset size (optional)
24244     root: "rows",                // The property which contains an Array of row objects
24245     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24246 }, RecordDef);
24247 </code></pre>
24248  * <p>
24249  * This would consume a JSON file like this:
24250  * <pre><code>
24251 { 'results': 2, 'rows': [
24252     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24253     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24254 }
24255 </code></pre>
24256  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24257  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24258  * paged from the remote server.
24259  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24260  * @cfg {String} root name of the property which contains the Array of row objects.
24261  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24262  * @cfg {Array} fields Array of field definition objects
24263  * @constructor
24264  * Create a new JsonReader
24265  * @param {Object} meta Metadata configuration options
24266  * @param {Object} recordType Either an Array of field definition objects,
24267  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24268  */
24269 Roo.data.JsonReader = function(meta, recordType){
24270     
24271     meta = meta || {};
24272     // set some defaults:
24273     Roo.applyIf(meta, {
24274         totalProperty: 'total',
24275         successProperty : 'success',
24276         root : 'data',
24277         id : 'id'
24278     });
24279     
24280     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24281 };
24282 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24283     
24284     /**
24285      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24286      * Used by Store query builder to append _requestMeta to params.
24287      * 
24288      */
24289     metaFromRemote : false,
24290     /**
24291      * This method is only used by a DataProxy which has retrieved data from a remote server.
24292      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24293      * @return {Object} data A data block which is used by an Roo.data.Store object as
24294      * a cache of Roo.data.Records.
24295      */
24296     read : function(response){
24297         var json = response.responseText;
24298        
24299         var o = /* eval:var:o */ eval("("+json+")");
24300         if(!o) {
24301             throw {message: "JsonReader.read: Json object not found"};
24302         }
24303         
24304         if(o.metaData){
24305             
24306             delete this.ef;
24307             this.metaFromRemote = true;
24308             this.meta = o.metaData;
24309             this.recordType = Roo.data.Record.create(o.metaData.fields);
24310             this.onMetaChange(this.meta, this.recordType, o);
24311         }
24312         return this.readRecords(o);
24313     },
24314
24315     // private function a store will implement
24316     onMetaChange : function(meta, recordType, o){
24317
24318     },
24319
24320     /**
24321          * @ignore
24322          */
24323     simpleAccess: function(obj, subsc) {
24324         return obj[subsc];
24325     },
24326
24327         /**
24328          * @ignore
24329          */
24330     getJsonAccessor: function(){
24331         var re = /[\[\.]/;
24332         return function(expr) {
24333             try {
24334                 return(re.test(expr))
24335                     ? new Function("obj", "return obj." + expr)
24336                     : function(obj){
24337                         return obj[expr];
24338                     };
24339             } catch(e){}
24340             return Roo.emptyFn;
24341         };
24342     }(),
24343
24344     /**
24345      * Create a data block containing Roo.data.Records from an XML document.
24346      * @param {Object} o An object which contains an Array of row objects in the property specified
24347      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24348      * which contains the total size of the dataset.
24349      * @return {Object} data A data block which is used by an Roo.data.Store object as
24350      * a cache of Roo.data.Records.
24351      */
24352     readRecords : function(o){
24353         /**
24354          * After any data loads, the raw JSON data is available for further custom processing.
24355          * @type Object
24356          */
24357         this.o = o;
24358         var s = this.meta, Record = this.recordType,
24359             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24360
24361 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24362         if (!this.ef) {
24363             if(s.totalProperty) {
24364                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24365                 }
24366                 if(s.successProperty) {
24367                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24368                 }
24369                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24370                 if (s.id) {
24371                         var g = this.getJsonAccessor(s.id);
24372                         this.getId = function(rec) {
24373                                 var r = g(rec);  
24374                                 return (r === undefined || r === "") ? null : r;
24375                         };
24376                 } else {
24377                         this.getId = function(){return null;};
24378                 }
24379             this.ef = [];
24380             for(var jj = 0; jj < fl; jj++){
24381                 f = fi[jj];
24382                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24383                 this.ef[jj] = this.getJsonAccessor(map);
24384             }
24385         }
24386
24387         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24388         if(s.totalProperty){
24389             var vt = parseInt(this.getTotal(o), 10);
24390             if(!isNaN(vt)){
24391                 totalRecords = vt;
24392             }
24393         }
24394         if(s.successProperty){
24395             var vs = this.getSuccess(o);
24396             if(vs === false || vs === 'false'){
24397                 success = false;
24398             }
24399         }
24400         var records = [];
24401         for(var i = 0; i < c; i++){
24402                 var n = root[i];
24403             var values = {};
24404             var id = this.getId(n);
24405             for(var j = 0; j < fl; j++){
24406                 f = fi[j];
24407             var v = this.ef[j](n);
24408             if (!f.convert) {
24409                 Roo.log('missing convert for ' + f.name);
24410                 Roo.log(f);
24411                 continue;
24412             }
24413             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24414             }
24415             var record = new Record(values, id);
24416             record.json = n;
24417             records[i] = record;
24418         }
24419         return {
24420             raw : o,
24421             success : success,
24422             records : records,
24423             totalRecords : totalRecords
24424         };
24425     }
24426 });/*
24427  * Based on:
24428  * Ext JS Library 1.1.1
24429  * Copyright(c) 2006-2007, Ext JS, LLC.
24430  *
24431  * Originally Released Under LGPL - original licence link has changed is not relivant.
24432  *
24433  * Fork - LGPL
24434  * <script type="text/javascript">
24435  */
24436
24437 /**
24438  * @class Roo.data.XmlReader
24439  * @extends Roo.data.DataReader
24440  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24441  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24442  * <p>
24443  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24444  * header in the HTTP response must be set to "text/xml".</em>
24445  * <p>
24446  * Example code:
24447  * <pre><code>
24448 var RecordDef = Roo.data.Record.create([
24449    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24450    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24451 ]);
24452 var myReader = new Roo.data.XmlReader({
24453    totalRecords: "results", // The element which contains the total dataset size (optional)
24454    record: "row",           // The repeated element which contains row information
24455    id: "id"                 // The element within the row that provides an ID for the record (optional)
24456 }, RecordDef);
24457 </code></pre>
24458  * <p>
24459  * This would consume an XML file like this:
24460  * <pre><code>
24461 &lt;?xml?>
24462 &lt;dataset>
24463  &lt;results>2&lt;/results>
24464  &lt;row>
24465    &lt;id>1&lt;/id>
24466    &lt;name>Bill&lt;/name>
24467    &lt;occupation>Gardener&lt;/occupation>
24468  &lt;/row>
24469  &lt;row>
24470    &lt;id>2&lt;/id>
24471    &lt;name>Ben&lt;/name>
24472    &lt;occupation>Horticulturalist&lt;/occupation>
24473  &lt;/row>
24474 &lt;/dataset>
24475 </code></pre>
24476  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24477  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24478  * paged from the remote server.
24479  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24480  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24481  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24482  * a record identifier value.
24483  * @constructor
24484  * Create a new XmlReader
24485  * @param {Object} meta Metadata configuration options
24486  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24487  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24488  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24489  */
24490 Roo.data.XmlReader = function(meta, recordType){
24491     meta = meta || {};
24492     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24493 };
24494 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24495     /**
24496      * This method is only used by a DataProxy which has retrieved data from a remote server.
24497          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24498          * to contain a method called 'responseXML' that returns an XML document object.
24499      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24500      * a cache of Roo.data.Records.
24501      */
24502     read : function(response){
24503         var doc = response.responseXML;
24504         if(!doc) {
24505             throw {message: "XmlReader.read: XML Document not available"};
24506         }
24507         return this.readRecords(doc);
24508     },
24509
24510     /**
24511      * Create a data block containing Roo.data.Records from an XML document.
24512          * @param {Object} doc A parsed XML document.
24513      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24514      * a cache of Roo.data.Records.
24515      */
24516     readRecords : function(doc){
24517         /**
24518          * After any data loads/reads, the raw XML Document is available for further custom processing.
24519          * @type XMLDocument
24520          */
24521         this.xmlData = doc;
24522         var root = doc.documentElement || doc;
24523         var q = Roo.DomQuery;
24524         var recordType = this.recordType, fields = recordType.prototype.fields;
24525         var sid = this.meta.id;
24526         var totalRecords = 0, success = true;
24527         if(this.meta.totalRecords){
24528             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24529         }
24530         
24531         if(this.meta.success){
24532             var sv = q.selectValue(this.meta.success, root, true);
24533             success = sv !== false && sv !== 'false';
24534         }
24535         var records = [];
24536         var ns = q.select(this.meta.record, root);
24537         for(var i = 0, len = ns.length; i < len; i++) {
24538                 var n = ns[i];
24539                 var values = {};
24540                 var id = sid ? q.selectValue(sid, n) : undefined;
24541                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24542                     var f = fields.items[j];
24543                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24544                     v = f.convert(v);
24545                     values[f.name] = v;
24546                 }
24547                 var record = new recordType(values, id);
24548                 record.node = n;
24549                 records[records.length] = record;
24550             }
24551
24552             return {
24553                 success : success,
24554                 records : records,
24555                 totalRecords : totalRecords || records.length
24556             };
24557     }
24558 });/*
24559  * Based on:
24560  * Ext JS Library 1.1.1
24561  * Copyright(c) 2006-2007, Ext JS, LLC.
24562  *
24563  * Originally Released Under LGPL - original licence link has changed is not relivant.
24564  *
24565  * Fork - LGPL
24566  * <script type="text/javascript">
24567  */
24568
24569 /**
24570  * @class Roo.data.ArrayReader
24571  * @extends Roo.data.DataReader
24572  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24573  * Each element of that Array represents a row of data fields. The
24574  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24575  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24576  * <p>
24577  * Example code:.
24578  * <pre><code>
24579 var RecordDef = Roo.data.Record.create([
24580     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24581     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24582 ]);
24583 var myReader = new Roo.data.ArrayReader({
24584     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24585 }, RecordDef);
24586 </code></pre>
24587  * <p>
24588  * This would consume an Array like this:
24589  * <pre><code>
24590 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24591   </code></pre>
24592  
24593  * @constructor
24594  * Create a new JsonReader
24595  * @param {Object} meta Metadata configuration options.
24596  * @param {Object|Array} recordType Either an Array of field definition objects
24597  * 
24598  * @cfg {Array} fields Array of field definition objects
24599  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24600  * as specified to {@link Roo.data.Record#create},
24601  * or an {@link Roo.data.Record} object
24602  *
24603  * 
24604  * created using {@link Roo.data.Record#create}.
24605  */
24606 Roo.data.ArrayReader = function(meta, recordType){
24607     
24608      
24609     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24610 };
24611
24612 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24613     /**
24614      * Create a data block containing Roo.data.Records from an XML document.
24615      * @param {Object} o An Array of row objects which represents the dataset.
24616      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24617      * a cache of Roo.data.Records.
24618      */
24619     readRecords : function(o){
24620         var sid = this.meta ? this.meta.id : null;
24621         var recordType = this.recordType, fields = recordType.prototype.fields;
24622         var records = [];
24623         var root = o;
24624             for(var i = 0; i < root.length; i++){
24625                     var n = root[i];
24626                 var values = {};
24627                 var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24628                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24629                 var f = fields.items[j];
24630                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24631                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24632                 v = f.convert(v);
24633                 values[f.name] = v;
24634             }
24635                 var record = new recordType(values, id);
24636                 record.json = n;
24637                 records[records.length] = record;
24638             }
24639             return {
24640                 records : records,
24641                 totalRecords : records.length
24642             };
24643     }
24644 });/*
24645  * Based on:
24646  * Ext JS Library 1.1.1
24647  * Copyright(c) 2006-2007, Ext JS, LLC.
24648  *
24649  * Originally Released Under LGPL - original licence link has changed is not relivant.
24650  *
24651  * Fork - LGPL
24652  * <script type="text/javascript">
24653  */
24654
24655
24656 /**
24657  * @class Roo.data.Tree
24658  * @extends Roo.util.Observable
24659  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24660  * in the tree have most standard DOM functionality.
24661  * @constructor
24662  * @param {Node} root (optional) The root node
24663  */
24664 Roo.data.Tree = function(root){
24665    this.nodeHash = {};
24666    /**
24667     * The root node for this tree
24668     * @type Node
24669     */
24670    this.root = null;
24671    if(root){
24672        this.setRootNode(root);
24673    }
24674    this.addEvents({
24675        /**
24676         * @event append
24677         * Fires when a new child node is appended to a node in this tree.
24678         * @param {Tree} tree The owner tree
24679         * @param {Node} parent The parent node
24680         * @param {Node} node The newly appended node
24681         * @param {Number} index The index of the newly appended node
24682         */
24683        "append" : true,
24684        /**
24685         * @event remove
24686         * Fires when a child node is removed from a node in this tree.
24687         * @param {Tree} tree The owner tree
24688         * @param {Node} parent The parent node
24689         * @param {Node} node The child node removed
24690         */
24691        "remove" : true,
24692        /**
24693         * @event move
24694         * Fires when a node is moved to a new location in the tree
24695         * @param {Tree} tree The owner tree
24696         * @param {Node} node The node moved
24697         * @param {Node} oldParent The old parent of this node
24698         * @param {Node} newParent The new parent of this node
24699         * @param {Number} index The index it was moved to
24700         */
24701        "move" : true,
24702        /**
24703         * @event insert
24704         * Fires when a new child node is inserted in a node in this tree.
24705         * @param {Tree} tree The owner tree
24706         * @param {Node} parent The parent node
24707         * @param {Node} node The child node inserted
24708         * @param {Node} refNode The child node the node was inserted before
24709         */
24710        "insert" : true,
24711        /**
24712         * @event beforeappend
24713         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24714         * @param {Tree} tree The owner tree
24715         * @param {Node} parent The parent node
24716         * @param {Node} node The child node to be appended
24717         */
24718        "beforeappend" : true,
24719        /**
24720         * @event beforeremove
24721         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24722         * @param {Tree} tree The owner tree
24723         * @param {Node} parent The parent node
24724         * @param {Node} node The child node to be removed
24725         */
24726        "beforeremove" : true,
24727        /**
24728         * @event beforemove
24729         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24730         * @param {Tree} tree The owner tree
24731         * @param {Node} node The node being moved
24732         * @param {Node} oldParent The parent of the node
24733         * @param {Node} newParent The new parent the node is moving to
24734         * @param {Number} index The index it is being moved to
24735         */
24736        "beforemove" : true,
24737        /**
24738         * @event beforeinsert
24739         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24740         * @param {Tree} tree The owner tree
24741         * @param {Node} parent The parent node
24742         * @param {Node} node The child node to be inserted
24743         * @param {Node} refNode The child node the node is being inserted before
24744         */
24745        "beforeinsert" : true
24746    });
24747
24748     Roo.data.Tree.superclass.constructor.call(this);
24749 };
24750
24751 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24752     pathSeparator: "/",
24753
24754     proxyNodeEvent : function(){
24755         return this.fireEvent.apply(this, arguments);
24756     },
24757
24758     /**
24759      * Returns the root node for this tree.
24760      * @return {Node}
24761      */
24762     getRootNode : function(){
24763         return this.root;
24764     },
24765
24766     /**
24767      * Sets the root node for this tree.
24768      * @param {Node} node
24769      * @return {Node}
24770      */
24771     setRootNode : function(node){
24772         this.root = node;
24773         node.ownerTree = this;
24774         node.isRoot = true;
24775         this.registerNode(node);
24776         return node;
24777     },
24778
24779     /**
24780      * Gets a node in this tree by its id.
24781      * @param {String} id
24782      * @return {Node}
24783      */
24784     getNodeById : function(id){
24785         return this.nodeHash[id];
24786     },
24787
24788     registerNode : function(node){
24789         this.nodeHash[node.id] = node;
24790     },
24791
24792     unregisterNode : function(node){
24793         delete this.nodeHash[node.id];
24794     },
24795
24796     toString : function(){
24797         return "[Tree"+(this.id?" "+this.id:"")+"]";
24798     }
24799 });
24800
24801 /**
24802  * @class Roo.data.Node
24803  * @extends Roo.util.Observable
24804  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24805  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24806  * @constructor
24807  * @param {Object} attributes The attributes/config for the node
24808  */
24809 Roo.data.Node = function(attributes){
24810     /**
24811      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24812      * @type {Object}
24813      */
24814     this.attributes = attributes || {};
24815     this.leaf = this.attributes.leaf;
24816     /**
24817      * The node id. @type String
24818      */
24819     this.id = this.attributes.id;
24820     if(!this.id){
24821         this.id = Roo.id(null, "ynode-");
24822         this.attributes.id = this.id;
24823     }
24824      
24825     
24826     /**
24827      * All child nodes of this node. @type Array
24828      */
24829     this.childNodes = [];
24830     if(!this.childNodes.indexOf){ // indexOf is a must
24831         this.childNodes.indexOf = function(o){
24832             for(var i = 0, len = this.length; i < len; i++){
24833                 if(this[i] == o) {
24834                     return i;
24835                 }
24836             }
24837             return -1;
24838         };
24839     }
24840     /**
24841      * The parent node for this node. @type Node
24842      */
24843     this.parentNode = null;
24844     /**
24845      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24846      */
24847     this.firstChild = null;
24848     /**
24849      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24850      */
24851     this.lastChild = null;
24852     /**
24853      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24854      */
24855     this.previousSibling = null;
24856     /**
24857      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24858      */
24859     this.nextSibling = null;
24860
24861     this.addEvents({
24862        /**
24863         * @event append
24864         * Fires when a new child node is appended
24865         * @param {Tree} tree The owner tree
24866         * @param {Node} this This node
24867         * @param {Node} node The newly appended node
24868         * @param {Number} index The index of the newly appended node
24869         */
24870        "append" : true,
24871        /**
24872         * @event remove
24873         * Fires when a child node is removed
24874         * @param {Tree} tree The owner tree
24875         * @param {Node} this This node
24876         * @param {Node} node The removed node
24877         */
24878        "remove" : true,
24879        /**
24880         * @event move
24881         * Fires when this node is moved to a new location in the tree
24882         * @param {Tree} tree The owner tree
24883         * @param {Node} this This node
24884         * @param {Node} oldParent The old parent of this node
24885         * @param {Node} newParent The new parent of this node
24886         * @param {Number} index The index it was moved to
24887         */
24888        "move" : true,
24889        /**
24890         * @event insert
24891         * Fires when a new child node is inserted.
24892         * @param {Tree} tree The owner tree
24893         * @param {Node} this This node
24894         * @param {Node} node The child node inserted
24895         * @param {Node} refNode The child node the node was inserted before
24896         */
24897        "insert" : true,
24898        /**
24899         * @event beforeappend
24900         * Fires before a new child is appended, return false to cancel the append.
24901         * @param {Tree} tree The owner tree
24902         * @param {Node} this This node
24903         * @param {Node} node The child node to be appended
24904         */
24905        "beforeappend" : true,
24906        /**
24907         * @event beforeremove
24908         * Fires before a child is removed, return false to cancel the remove.
24909         * @param {Tree} tree The owner tree
24910         * @param {Node} this This node
24911         * @param {Node} node The child node to be removed
24912         */
24913        "beforeremove" : true,
24914        /**
24915         * @event beforemove
24916         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24917         * @param {Tree} tree The owner tree
24918         * @param {Node} this This node
24919         * @param {Node} oldParent The parent of this node
24920         * @param {Node} newParent The new parent this node is moving to
24921         * @param {Number} index The index it is being moved to
24922         */
24923        "beforemove" : true,
24924        /**
24925         * @event beforeinsert
24926         * Fires before a new child is inserted, return false to cancel the insert.
24927         * @param {Tree} tree The owner tree
24928         * @param {Node} this This node
24929         * @param {Node} node The child node to be inserted
24930         * @param {Node} refNode The child node the node is being inserted before
24931         */
24932        "beforeinsert" : true
24933    });
24934     this.listeners = this.attributes.listeners;
24935     Roo.data.Node.superclass.constructor.call(this);
24936 };
24937
24938 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24939     fireEvent : function(evtName){
24940         // first do standard event for this node
24941         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24942             return false;
24943         }
24944         // then bubble it up to the tree if the event wasn't cancelled
24945         var ot = this.getOwnerTree();
24946         if(ot){
24947             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24948                 return false;
24949             }
24950         }
24951         return true;
24952     },
24953
24954     /**
24955      * Returns true if this node is a leaf
24956      * @return {Boolean}
24957      */
24958     isLeaf : function(){
24959         return this.leaf === true;
24960     },
24961
24962     // private
24963     setFirstChild : function(node){
24964         this.firstChild = node;
24965     },
24966
24967     //private
24968     setLastChild : function(node){
24969         this.lastChild = node;
24970     },
24971
24972
24973     /**
24974      * Returns true if this node is the last child of its parent
24975      * @return {Boolean}
24976      */
24977     isLast : function(){
24978        return (!this.parentNode ? true : this.parentNode.lastChild == this);
24979     },
24980
24981     /**
24982      * Returns true if this node is the first child of its parent
24983      * @return {Boolean}
24984      */
24985     isFirst : function(){
24986        return (!this.parentNode ? true : this.parentNode.firstChild == this);
24987     },
24988
24989     hasChildNodes : function(){
24990         return !this.isLeaf() && this.childNodes.length > 0;
24991     },
24992
24993     /**
24994      * Insert node(s) as the last child node of this node.
24995      * @param {Node/Array} node The node or Array of nodes to append
24996      * @return {Node} The appended node if single append, or null if an array was passed
24997      */
24998     appendChild : function(node){
24999         var multi = false;
25000         if(node instanceof Array){
25001             multi = node;
25002         }else if(arguments.length > 1){
25003             multi = arguments;
25004         }
25005         
25006         // if passed an array or multiple args do them one by one
25007         if(multi){
25008             for(var i = 0, len = multi.length; i < len; i++) {
25009                 this.appendChild(multi[i]);
25010             }
25011         }else{
25012             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25013                 return false;
25014             }
25015             var index = this.childNodes.length;
25016             var oldParent = node.parentNode;
25017             // it's a move, make sure we move it cleanly
25018             if(oldParent){
25019                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25020                     return false;
25021                 }
25022                 oldParent.removeChild(node);
25023             }
25024             
25025             index = this.childNodes.length;
25026             if(index == 0){
25027                 this.setFirstChild(node);
25028             }
25029             this.childNodes.push(node);
25030             node.parentNode = this;
25031             var ps = this.childNodes[index-1];
25032             if(ps){
25033                 node.previousSibling = ps;
25034                 ps.nextSibling = node;
25035             }else{
25036                 node.previousSibling = null;
25037             }
25038             node.nextSibling = null;
25039             this.setLastChild(node);
25040             node.setOwnerTree(this.getOwnerTree());
25041             this.fireEvent("append", this.ownerTree, this, node, index);
25042             if(this.ownerTree) {
25043                 this.ownerTree.fireEvent("appendnode", this, node, index);
25044             }
25045             if(oldParent){
25046                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25047             }
25048             return node;
25049         }
25050     },
25051
25052     /**
25053      * Removes a child node from this node.
25054      * @param {Node} node The node to remove
25055      * @return {Node} The removed node
25056      */
25057     removeChild : function(node){
25058         var index = this.childNodes.indexOf(node);
25059         if(index == -1){
25060             return false;
25061         }
25062         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25063             return false;
25064         }
25065
25066         // remove it from childNodes collection
25067         this.childNodes.splice(index, 1);
25068
25069         // update siblings
25070         if(node.previousSibling){
25071             node.previousSibling.nextSibling = node.nextSibling;
25072         }
25073         if(node.nextSibling){
25074             node.nextSibling.previousSibling = node.previousSibling;
25075         }
25076
25077         // update child refs
25078         if(this.firstChild == node){
25079             this.setFirstChild(node.nextSibling);
25080         }
25081         if(this.lastChild == node){
25082             this.setLastChild(node.previousSibling);
25083         }
25084
25085         node.setOwnerTree(null);
25086         // clear any references from the node
25087         node.parentNode = null;
25088         node.previousSibling = null;
25089         node.nextSibling = null;
25090         this.fireEvent("remove", this.ownerTree, this, node);
25091         return node;
25092     },
25093
25094     /**
25095      * Inserts the first node before the second node in this nodes childNodes collection.
25096      * @param {Node} node The node to insert
25097      * @param {Node} refNode The node to insert before (if null the node is appended)
25098      * @return {Node} The inserted node
25099      */
25100     insertBefore : function(node, refNode){
25101         if(!refNode){ // like standard Dom, refNode can be null for append
25102             return this.appendChild(node);
25103         }
25104         // nothing to do
25105         if(node == refNode){
25106             return false;
25107         }
25108
25109         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25110             return false;
25111         }
25112         var index = this.childNodes.indexOf(refNode);
25113         var oldParent = node.parentNode;
25114         var refIndex = index;
25115
25116         // when moving internally, indexes will change after remove
25117         if(oldParent == this && this.childNodes.indexOf(node) < index){
25118             refIndex--;
25119         }
25120
25121         // it's a move, make sure we move it cleanly
25122         if(oldParent){
25123             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25124                 return false;
25125             }
25126             oldParent.removeChild(node);
25127         }
25128         if(refIndex == 0){
25129             this.setFirstChild(node);
25130         }
25131         this.childNodes.splice(refIndex, 0, node);
25132         node.parentNode = this;
25133         var ps = this.childNodes[refIndex-1];
25134         if(ps){
25135             node.previousSibling = ps;
25136             ps.nextSibling = node;
25137         }else{
25138             node.previousSibling = null;
25139         }
25140         node.nextSibling = refNode;
25141         refNode.previousSibling = node;
25142         node.setOwnerTree(this.getOwnerTree());
25143         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25144         if(oldParent){
25145             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25146         }
25147         return node;
25148     },
25149
25150     /**
25151      * Returns the child node at the specified index.
25152      * @param {Number} index
25153      * @return {Node}
25154      */
25155     item : function(index){
25156         return this.childNodes[index];
25157     },
25158
25159     /**
25160      * Replaces one child node in this node with another.
25161      * @param {Node} newChild The replacement node
25162      * @param {Node} oldChild The node to replace
25163      * @return {Node} The replaced node
25164      */
25165     replaceChild : function(newChild, oldChild){
25166         this.insertBefore(newChild, oldChild);
25167         this.removeChild(oldChild);
25168         return oldChild;
25169     },
25170
25171     /**
25172      * Returns the index of a child node
25173      * @param {Node} node
25174      * @return {Number} The index of the node or -1 if it was not found
25175      */
25176     indexOf : function(child){
25177         return this.childNodes.indexOf(child);
25178     },
25179
25180     /**
25181      * Returns the tree this node is in.
25182      * @return {Tree}
25183      */
25184     getOwnerTree : function(){
25185         // if it doesn't have one, look for one
25186         if(!this.ownerTree){
25187             var p = this;
25188             while(p){
25189                 if(p.ownerTree){
25190                     this.ownerTree = p.ownerTree;
25191                     break;
25192                 }
25193                 p = p.parentNode;
25194             }
25195         }
25196         return this.ownerTree;
25197     },
25198
25199     /**
25200      * Returns depth of this node (the root node has a depth of 0)
25201      * @return {Number}
25202      */
25203     getDepth : function(){
25204         var depth = 0;
25205         var p = this;
25206         while(p.parentNode){
25207             ++depth;
25208             p = p.parentNode;
25209         }
25210         return depth;
25211     },
25212
25213     // private
25214     setOwnerTree : function(tree){
25215         // if it's move, we need to update everyone
25216         if(tree != this.ownerTree){
25217             if(this.ownerTree){
25218                 this.ownerTree.unregisterNode(this);
25219             }
25220             this.ownerTree = tree;
25221             var cs = this.childNodes;
25222             for(var i = 0, len = cs.length; i < len; i++) {
25223                 cs[i].setOwnerTree(tree);
25224             }
25225             if(tree){
25226                 tree.registerNode(this);
25227             }
25228         }
25229     },
25230
25231     /**
25232      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25233      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25234      * @return {String} The path
25235      */
25236     getPath : function(attr){
25237         attr = attr || "id";
25238         var p = this.parentNode;
25239         var b = [this.attributes[attr]];
25240         while(p){
25241             b.unshift(p.attributes[attr]);
25242             p = p.parentNode;
25243         }
25244         var sep = this.getOwnerTree().pathSeparator;
25245         return sep + b.join(sep);
25246     },
25247
25248     /**
25249      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25250      * function call will be the scope provided or the current node. The arguments to the function
25251      * will be the args provided or the current node. If the function returns false at any point,
25252      * the bubble is stopped.
25253      * @param {Function} fn The function to call
25254      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25255      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25256      */
25257     bubble : function(fn, scope, args){
25258         var p = this;
25259         while(p){
25260             if(fn.call(scope || p, args || p) === false){
25261                 break;
25262             }
25263             p = p.parentNode;
25264         }
25265     },
25266
25267     /**
25268      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25269      * function call will be the scope provided or the current node. The arguments to the function
25270      * will be the args provided or the current node. If the function returns false at any point,
25271      * the cascade is stopped on that branch.
25272      * @param {Function} fn The function to call
25273      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25274      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25275      */
25276     cascade : function(fn, scope, args){
25277         if(fn.call(scope || this, args || this) !== false){
25278             var cs = this.childNodes;
25279             for(var i = 0, len = cs.length; i < len; i++) {
25280                 cs[i].cascade(fn, scope, args);
25281             }
25282         }
25283     },
25284
25285     /**
25286      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25287      * function call will be the scope provided or the current node. The arguments to the function
25288      * will be the args provided or the current node. If the function returns false at any point,
25289      * the iteration stops.
25290      * @param {Function} fn The function to call
25291      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25292      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25293      */
25294     eachChild : function(fn, scope, args){
25295         var cs = this.childNodes;
25296         for(var i = 0, len = cs.length; i < len; i++) {
25297                 if(fn.call(scope || this, args || cs[i]) === false){
25298                     break;
25299                 }
25300         }
25301     },
25302
25303     /**
25304      * Finds the first child that has the attribute with the specified value.
25305      * @param {String} attribute The attribute name
25306      * @param {Mixed} value The value to search for
25307      * @return {Node} The found child or null if none was found
25308      */
25309     findChild : function(attribute, value){
25310         var cs = this.childNodes;
25311         for(var i = 0, len = cs.length; i < len; i++) {
25312                 if(cs[i].attributes[attribute] == value){
25313                     return cs[i];
25314                 }
25315         }
25316         return null;
25317     },
25318
25319     /**
25320      * Finds the first child by a custom function. The child matches if the function passed
25321      * returns true.
25322      * @param {Function} fn
25323      * @param {Object} scope (optional)
25324      * @return {Node} The found child or null if none was found
25325      */
25326     findChildBy : function(fn, scope){
25327         var cs = this.childNodes;
25328         for(var i = 0, len = cs.length; i < len; i++) {
25329                 if(fn.call(scope||cs[i], cs[i]) === true){
25330                     return cs[i];
25331                 }
25332         }
25333         return null;
25334     },
25335
25336     /**
25337      * Sorts this nodes children using the supplied sort function
25338      * @param {Function} fn
25339      * @param {Object} scope (optional)
25340      */
25341     sort : function(fn, scope){
25342         var cs = this.childNodes;
25343         var len = cs.length;
25344         if(len > 0){
25345             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25346             cs.sort(sortFn);
25347             for(var i = 0; i < len; i++){
25348                 var n = cs[i];
25349                 n.previousSibling = cs[i-1];
25350                 n.nextSibling = cs[i+1];
25351                 if(i == 0){
25352                     this.setFirstChild(n);
25353                 }
25354                 if(i == len-1){
25355                     this.setLastChild(n);
25356                 }
25357             }
25358         }
25359     },
25360
25361     /**
25362      * Returns true if this node is an ancestor (at any point) of the passed node.
25363      * @param {Node} node
25364      * @return {Boolean}
25365      */
25366     contains : function(node){
25367         return node.isAncestor(this);
25368     },
25369
25370     /**
25371      * Returns true if the passed node is an ancestor (at any point) of this node.
25372      * @param {Node} node
25373      * @return {Boolean}
25374      */
25375     isAncestor : function(node){
25376         var p = this.parentNode;
25377         while(p){
25378             if(p == node){
25379                 return true;
25380             }
25381             p = p.parentNode;
25382         }
25383         return false;
25384     },
25385
25386     toString : function(){
25387         return "[Node"+(this.id?" "+this.id:"")+"]";
25388     }
25389 });/*
25390  * Based on:
25391  * Ext JS Library 1.1.1
25392  * Copyright(c) 2006-2007, Ext JS, LLC.
25393  *
25394  * Originally Released Under LGPL - original licence link has changed is not relivant.
25395  *
25396  * Fork - LGPL
25397  * <script type="text/javascript">
25398  */
25399  (function(){ 
25400 /**
25401  * @class Roo.Layer
25402  * @extends Roo.Element
25403  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25404  * automatic maintaining of shadow/shim positions.
25405  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25406  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25407  * you can pass a string with a CSS class name. False turns off the shadow.
25408  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25409  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25410  * @cfg {String} cls CSS class to add to the element
25411  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25412  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25413  * @constructor
25414  * @param {Object} config An object with config options.
25415  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25416  */
25417
25418 Roo.Layer = function(config, existingEl){
25419     config = config || {};
25420     var dh = Roo.DomHelper;
25421     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25422     if(existingEl){
25423         this.dom = Roo.getDom(existingEl);
25424     }
25425     if(!this.dom){
25426         var o = config.dh || {tag: "div", cls: "x-layer"};
25427         this.dom = dh.append(pel, o);
25428     }
25429     if(config.cls){
25430         this.addClass(config.cls);
25431     }
25432     this.constrain = config.constrain !== false;
25433     this.visibilityMode = Roo.Element.VISIBILITY;
25434     if(config.id){
25435         this.id = this.dom.id = config.id;
25436     }else{
25437         this.id = Roo.id(this.dom);
25438     }
25439     this.zindex = config.zindex || this.getZIndex();
25440     this.position("absolute", this.zindex);
25441     if(config.shadow){
25442         this.shadowOffset = config.shadowOffset || 4;
25443         this.shadow = new Roo.Shadow({
25444             offset : this.shadowOffset,
25445             mode : config.shadow
25446         });
25447     }else{
25448         this.shadowOffset = 0;
25449     }
25450     this.useShim = config.shim !== false && Roo.useShims;
25451     this.useDisplay = config.useDisplay;
25452     this.hide();
25453 };
25454
25455 var supr = Roo.Element.prototype;
25456
25457 // shims are shared among layer to keep from having 100 iframes
25458 var shims = [];
25459
25460 Roo.extend(Roo.Layer, Roo.Element, {
25461
25462     getZIndex : function(){
25463         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25464     },
25465
25466     getShim : function(){
25467         if(!this.useShim){
25468             return null;
25469         }
25470         if(this.shim){
25471             return this.shim;
25472         }
25473         var shim = shims.shift();
25474         if(!shim){
25475             shim = this.createShim();
25476             shim.enableDisplayMode('block');
25477             shim.dom.style.display = 'none';
25478             shim.dom.style.visibility = 'visible';
25479         }
25480         var pn = this.dom.parentNode;
25481         if(shim.dom.parentNode != pn){
25482             pn.insertBefore(shim.dom, this.dom);
25483         }
25484         shim.setStyle('z-index', this.getZIndex()-2);
25485         this.shim = shim;
25486         return shim;
25487     },
25488
25489     hideShim : function(){
25490         if(this.shim){
25491             this.shim.setDisplayed(false);
25492             shims.push(this.shim);
25493             delete this.shim;
25494         }
25495     },
25496
25497     disableShadow : function(){
25498         if(this.shadow){
25499             this.shadowDisabled = true;
25500             this.shadow.hide();
25501             this.lastShadowOffset = this.shadowOffset;
25502             this.shadowOffset = 0;
25503         }
25504     },
25505
25506     enableShadow : function(show){
25507         if(this.shadow){
25508             this.shadowDisabled = false;
25509             this.shadowOffset = this.lastShadowOffset;
25510             delete this.lastShadowOffset;
25511             if(show){
25512                 this.sync(true);
25513             }
25514         }
25515     },
25516
25517     // private
25518     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25519     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25520     sync : function(doShow){
25521         var sw = this.shadow;
25522         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25523             var sh = this.getShim();
25524
25525             var w = this.getWidth(),
25526                 h = this.getHeight();
25527
25528             var l = this.getLeft(true),
25529                 t = this.getTop(true);
25530
25531             if(sw && !this.shadowDisabled){
25532                 if(doShow && !sw.isVisible()){
25533                     sw.show(this);
25534                 }else{
25535                     sw.realign(l, t, w, h);
25536                 }
25537                 if(sh){
25538                     if(doShow){
25539                        sh.show();
25540                     }
25541                     // fit the shim behind the shadow, so it is shimmed too
25542                     var a = sw.adjusts, s = sh.dom.style;
25543                     s.left = (Math.min(l, l+a.l))+"px";
25544                     s.top = (Math.min(t, t+a.t))+"px";
25545                     s.width = (w+a.w)+"px";
25546                     s.height = (h+a.h)+"px";
25547                 }
25548             }else if(sh){
25549                 if(doShow){
25550                    sh.show();
25551                 }
25552                 sh.setSize(w, h);
25553                 sh.setLeftTop(l, t);
25554             }
25555             
25556         }
25557     },
25558
25559     // private
25560     destroy : function(){
25561         this.hideShim();
25562         if(this.shadow){
25563             this.shadow.hide();
25564         }
25565         this.removeAllListeners();
25566         var pn = this.dom.parentNode;
25567         if(pn){
25568             pn.removeChild(this.dom);
25569         }
25570         Roo.Element.uncache(this.id);
25571     },
25572
25573     remove : function(){
25574         this.destroy();
25575     },
25576
25577     // private
25578     beginUpdate : function(){
25579         this.updating = true;
25580     },
25581
25582     // private
25583     endUpdate : function(){
25584         this.updating = false;
25585         this.sync(true);
25586     },
25587
25588     // private
25589     hideUnders : function(negOffset){
25590         if(this.shadow){
25591             this.shadow.hide();
25592         }
25593         this.hideShim();
25594     },
25595
25596     // private
25597     constrainXY : function(){
25598         if(this.constrain){
25599             var vw = Roo.lib.Dom.getViewWidth(),
25600                 vh = Roo.lib.Dom.getViewHeight();
25601             var s = Roo.get(document).getScroll();
25602
25603             var xy = this.getXY();
25604             var x = xy[0], y = xy[1];   
25605             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25606             // only move it if it needs it
25607             var moved = false;
25608             // first validate right/bottom
25609             if((x + w) > vw+s.left){
25610                 x = vw - w - this.shadowOffset;
25611                 moved = true;
25612             }
25613             if((y + h) > vh+s.top){
25614                 y = vh - h - this.shadowOffset;
25615                 moved = true;
25616             }
25617             // then make sure top/left isn't negative
25618             if(x < s.left){
25619                 x = s.left;
25620                 moved = true;
25621             }
25622             if(y < s.top){
25623                 y = s.top;
25624                 moved = true;
25625             }
25626             if(moved){
25627                 if(this.avoidY){
25628                     var ay = this.avoidY;
25629                     if(y <= ay && (y+h) >= ay){
25630                         y = ay-h-5;   
25631                     }
25632                 }
25633                 xy = [x, y];
25634                 this.storeXY(xy);
25635                 supr.setXY.call(this, xy);
25636                 this.sync();
25637             }
25638         }
25639     },
25640
25641     isVisible : function(){
25642         return this.visible;    
25643     },
25644
25645     // private
25646     showAction : function(){
25647         this.visible = true; // track visibility to prevent getStyle calls
25648         if(this.useDisplay === true){
25649             this.setDisplayed("");
25650         }else if(this.lastXY){
25651             supr.setXY.call(this, this.lastXY);
25652         }else if(this.lastLT){
25653             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25654         }
25655     },
25656
25657     // private
25658     hideAction : function(){
25659         this.visible = false;
25660         if(this.useDisplay === true){
25661             this.setDisplayed(false);
25662         }else{
25663             this.setLeftTop(-10000,-10000);
25664         }
25665     },
25666
25667     // overridden Element method
25668     setVisible : function(v, a, d, c, e){
25669         if(v){
25670             this.showAction();
25671         }
25672         if(a && v){
25673             var cb = function(){
25674                 this.sync(true);
25675                 if(c){
25676                     c();
25677                 }
25678             }.createDelegate(this);
25679             supr.setVisible.call(this, true, true, d, cb, e);
25680         }else{
25681             if(!v){
25682                 this.hideUnders(true);
25683             }
25684             var cb = c;
25685             if(a){
25686                 cb = function(){
25687                     this.hideAction();
25688                     if(c){
25689                         c();
25690                     }
25691                 }.createDelegate(this);
25692             }
25693             supr.setVisible.call(this, v, a, d, cb, e);
25694             if(v){
25695                 this.sync(true);
25696             }else if(!a){
25697                 this.hideAction();
25698             }
25699         }
25700     },
25701
25702     storeXY : function(xy){
25703         delete this.lastLT;
25704         this.lastXY = xy;
25705     },
25706
25707     storeLeftTop : function(left, top){
25708         delete this.lastXY;
25709         this.lastLT = [left, top];
25710     },
25711
25712     // private
25713     beforeFx : function(){
25714         this.beforeAction();
25715         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25716     },
25717
25718     // private
25719     afterFx : function(){
25720         Roo.Layer.superclass.afterFx.apply(this, arguments);
25721         this.sync(this.isVisible());
25722     },
25723
25724     // private
25725     beforeAction : function(){
25726         if(!this.updating && this.shadow){
25727             this.shadow.hide();
25728         }
25729     },
25730
25731     // overridden Element method
25732     setLeft : function(left){
25733         this.storeLeftTop(left, this.getTop(true));
25734         supr.setLeft.apply(this, arguments);
25735         this.sync();
25736     },
25737
25738     setTop : function(top){
25739         this.storeLeftTop(this.getLeft(true), top);
25740         supr.setTop.apply(this, arguments);
25741         this.sync();
25742     },
25743
25744     setLeftTop : function(left, top){
25745         this.storeLeftTop(left, top);
25746         supr.setLeftTop.apply(this, arguments);
25747         this.sync();
25748     },
25749
25750     setXY : function(xy, a, d, c, e){
25751         this.fixDisplay();
25752         this.beforeAction();
25753         this.storeXY(xy);
25754         var cb = this.createCB(c);
25755         supr.setXY.call(this, xy, a, d, cb, e);
25756         if(!a){
25757             cb();
25758         }
25759     },
25760
25761     // private
25762     createCB : function(c){
25763         var el = this;
25764         return function(){
25765             el.constrainXY();
25766             el.sync(true);
25767             if(c){
25768                 c();
25769             }
25770         };
25771     },
25772
25773     // overridden Element method
25774     setX : function(x, a, d, c, e){
25775         this.setXY([x, this.getY()], a, d, c, e);
25776     },
25777
25778     // overridden Element method
25779     setY : function(y, a, d, c, e){
25780         this.setXY([this.getX(), y], a, d, c, e);
25781     },
25782
25783     // overridden Element method
25784     setSize : function(w, h, a, d, c, e){
25785         this.beforeAction();
25786         var cb = this.createCB(c);
25787         supr.setSize.call(this, w, h, a, d, cb, e);
25788         if(!a){
25789             cb();
25790         }
25791     },
25792
25793     // overridden Element method
25794     setWidth : function(w, a, d, c, e){
25795         this.beforeAction();
25796         var cb = this.createCB(c);
25797         supr.setWidth.call(this, w, a, d, cb, e);
25798         if(!a){
25799             cb();
25800         }
25801     },
25802
25803     // overridden Element method
25804     setHeight : function(h, a, d, c, e){
25805         this.beforeAction();
25806         var cb = this.createCB(c);
25807         supr.setHeight.call(this, h, a, d, cb, e);
25808         if(!a){
25809             cb();
25810         }
25811     },
25812
25813     // overridden Element method
25814     setBounds : function(x, y, w, h, a, d, c, e){
25815         this.beforeAction();
25816         var cb = this.createCB(c);
25817         if(!a){
25818             this.storeXY([x, y]);
25819             supr.setXY.call(this, [x, y]);
25820             supr.setSize.call(this, w, h, a, d, cb, e);
25821             cb();
25822         }else{
25823             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25824         }
25825         return this;
25826     },
25827     
25828     /**
25829      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25830      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25831      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25832      * @param {Number} zindex The new z-index to set
25833      * @return {this} The Layer
25834      */
25835     setZIndex : function(zindex){
25836         this.zindex = zindex;
25837         this.setStyle("z-index", zindex + 2);
25838         if(this.shadow){
25839             this.shadow.setZIndex(zindex + 1);
25840         }
25841         if(this.shim){
25842             this.shim.setStyle("z-index", zindex);
25843         }
25844     }
25845 });
25846 })();/*
25847  * Based on:
25848  * Ext JS Library 1.1.1
25849  * Copyright(c) 2006-2007, Ext JS, LLC.
25850  *
25851  * Originally Released Under LGPL - original licence link has changed is not relivant.
25852  *
25853  * Fork - LGPL
25854  * <script type="text/javascript">
25855  */
25856
25857
25858 /**
25859  * @class Roo.Shadow
25860  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25861  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25862  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25863  * @constructor
25864  * Create a new Shadow
25865  * @param {Object} config The config object
25866  */
25867 Roo.Shadow = function(config){
25868     Roo.apply(this, config);
25869     if(typeof this.mode != "string"){
25870         this.mode = this.defaultMode;
25871     }
25872     var o = this.offset, a = {h: 0};
25873     var rad = Math.floor(this.offset/2);
25874     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25875         case "drop":
25876             a.w = 0;
25877             a.l = a.t = o;
25878             a.t -= 1;
25879             if(Roo.isIE){
25880                 a.l -= this.offset + rad;
25881                 a.t -= this.offset + rad;
25882                 a.w -= rad;
25883                 a.h -= rad;
25884                 a.t += 1;
25885             }
25886         break;
25887         case "sides":
25888             a.w = (o*2);
25889             a.l = -o;
25890             a.t = o-1;
25891             if(Roo.isIE){
25892                 a.l -= (this.offset - rad);
25893                 a.t -= this.offset + rad;
25894                 a.l += 1;
25895                 a.w -= (this.offset - rad)*2;
25896                 a.w -= rad + 1;
25897                 a.h -= 1;
25898             }
25899         break;
25900         case "frame":
25901             a.w = a.h = (o*2);
25902             a.l = a.t = -o;
25903             a.t += 1;
25904             a.h -= 2;
25905             if(Roo.isIE){
25906                 a.l -= (this.offset - rad);
25907                 a.t -= (this.offset - rad);
25908                 a.l += 1;
25909                 a.w -= (this.offset + rad + 1);
25910                 a.h -= (this.offset + rad);
25911                 a.h += 1;
25912             }
25913         break;
25914     };
25915
25916     this.adjusts = a;
25917 };
25918
25919 Roo.Shadow.prototype = {
25920     /**
25921      * @cfg {String} mode
25922      * The shadow display mode.  Supports the following options:<br />
25923      * sides: Shadow displays on both sides and bottom only<br />
25924      * frame: Shadow displays equally on all four sides<br />
25925      * drop: Traditional bottom-right drop shadow (default)
25926      */
25927     /**
25928      * @cfg {String} offset
25929      * The number of pixels to offset the shadow from the element (defaults to 4)
25930      */
25931     offset: 4,
25932
25933     // private
25934     defaultMode: "drop",
25935
25936     /**
25937      * Displays the shadow under the target element
25938      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25939      */
25940     show : function(target){
25941         target = Roo.get(target);
25942         if(!this.el){
25943             this.el = Roo.Shadow.Pool.pull();
25944             if(this.el.dom.nextSibling != target.dom){
25945                 this.el.insertBefore(target);
25946             }
25947         }
25948         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25949         if(Roo.isIE){
25950             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25951         }
25952         this.realign(
25953             target.getLeft(true),
25954             target.getTop(true),
25955             target.getWidth(),
25956             target.getHeight()
25957         );
25958         this.el.dom.style.display = "block";
25959     },
25960
25961     /**
25962      * Returns true if the shadow is visible, else false
25963      */
25964     isVisible : function(){
25965         return this.el ? true : false;  
25966     },
25967
25968     /**
25969      * Direct alignment when values are already available. Show must be called at least once before
25970      * calling this method to ensure it is initialized.
25971      * @param {Number} left The target element left position
25972      * @param {Number} top The target element top position
25973      * @param {Number} width The target element width
25974      * @param {Number} height The target element height
25975      */
25976     realign : function(l, t, w, h){
25977         if(!this.el){
25978             return;
25979         }
25980         var a = this.adjusts, d = this.el.dom, s = d.style;
25981         var iea = 0;
25982         s.left = (l+a.l)+"px";
25983         s.top = (t+a.t)+"px";
25984         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
25985  
25986         if(s.width != sws || s.height != shs){
25987             s.width = sws;
25988             s.height = shs;
25989             if(!Roo.isIE){
25990                 var cn = d.childNodes;
25991                 var sww = Math.max(0, (sw-12))+"px";
25992                 cn[0].childNodes[1].style.width = sww;
25993                 cn[1].childNodes[1].style.width = sww;
25994                 cn[2].childNodes[1].style.width = sww;
25995                 cn[1].style.height = Math.max(0, (sh-12))+"px";
25996             }
25997         }
25998     },
25999
26000     /**
26001      * Hides this shadow
26002      */
26003     hide : function(){
26004         if(this.el){
26005             this.el.dom.style.display = "none";
26006             Roo.Shadow.Pool.push(this.el);
26007             delete this.el;
26008         }
26009     },
26010
26011     /**
26012      * Adjust the z-index of this shadow
26013      * @param {Number} zindex The new z-index
26014      */
26015     setZIndex : function(z){
26016         this.zIndex = z;
26017         if(this.el){
26018             this.el.setStyle("z-index", z);
26019         }
26020     }
26021 };
26022
26023 // Private utility class that manages the internal Shadow cache
26024 Roo.Shadow.Pool = function(){
26025     var p = [];
26026     var markup = Roo.isIE ?
26027                  '<div class="x-ie-shadow"></div>' :
26028                  '<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>';
26029     return {
26030         pull : function(){
26031             var sh = p.shift();
26032             if(!sh){
26033                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26034                 sh.autoBoxAdjust = false;
26035             }
26036             return sh;
26037         },
26038
26039         push : function(sh){
26040             p.push(sh);
26041         }
26042     };
26043 }();/*
26044  * Based on:
26045  * Ext JS Library 1.1.1
26046  * Copyright(c) 2006-2007, Ext JS, LLC.
26047  *
26048  * Originally Released Under LGPL - original licence link has changed is not relivant.
26049  *
26050  * Fork - LGPL
26051  * <script type="text/javascript">
26052  */
26053
26054
26055 /**
26056  * @class Roo.SplitBar
26057  * @extends Roo.util.Observable
26058  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26059  * <br><br>
26060  * Usage:
26061  * <pre><code>
26062 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26063                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26064 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26065 split.minSize = 100;
26066 split.maxSize = 600;
26067 split.animate = true;
26068 split.on('moved', splitterMoved);
26069 </code></pre>
26070  * @constructor
26071  * Create a new SplitBar
26072  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26073  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26074  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26075  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26076                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26077                         position of the SplitBar).
26078  */
26079 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26080     
26081     /** @private */
26082     this.el = Roo.get(dragElement, true);
26083     this.el.dom.unselectable = "on";
26084     /** @private */
26085     this.resizingEl = Roo.get(resizingElement, true);
26086
26087     /**
26088      * @private
26089      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26090      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26091      * @type Number
26092      */
26093     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26094     
26095     /**
26096      * The minimum size of the resizing element. (Defaults to 0)
26097      * @type Number
26098      */
26099     this.minSize = 0;
26100     
26101     /**
26102      * The maximum size of the resizing element. (Defaults to 2000)
26103      * @type Number
26104      */
26105     this.maxSize = 2000;
26106     
26107     /**
26108      * Whether to animate the transition to the new size
26109      * @type Boolean
26110      */
26111     this.animate = false;
26112     
26113     /**
26114      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26115      * @type Boolean
26116      */
26117     this.useShim = false;
26118     
26119     /** @private */
26120     this.shim = null;
26121     
26122     if(!existingProxy){
26123         /** @private */
26124         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26125     }else{
26126         this.proxy = Roo.get(existingProxy).dom;
26127     }
26128     /** @private */
26129     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26130     
26131     /** @private */
26132     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26133     
26134     /** @private */
26135     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26136     
26137     /** @private */
26138     this.dragSpecs = {};
26139     
26140     /**
26141      * @private The adapter to use to positon and resize elements
26142      */
26143     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26144     this.adapter.init(this);
26145     
26146     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26147         /** @private */
26148         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26149         this.el.addClass("x-splitbar-h");
26150     }else{
26151         /** @private */
26152         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26153         this.el.addClass("x-splitbar-v");
26154     }
26155     
26156     this.addEvents({
26157         /**
26158          * @event resize
26159          * Fires when the splitter is moved (alias for {@link #event-moved})
26160          * @param {Roo.SplitBar} this
26161          * @param {Number} newSize the new width or height
26162          */
26163         "resize" : true,
26164         /**
26165          * @event moved
26166          * Fires when the splitter is moved
26167          * @param {Roo.SplitBar} this
26168          * @param {Number} newSize the new width or height
26169          */
26170         "moved" : true,
26171         /**
26172          * @event beforeresize
26173          * Fires before the splitter is dragged
26174          * @param {Roo.SplitBar} this
26175          */
26176         "beforeresize" : true,
26177
26178         "beforeapply" : true
26179     });
26180
26181     Roo.util.Observable.call(this);
26182 };
26183
26184 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26185     onStartProxyDrag : function(x, y){
26186         this.fireEvent("beforeresize", this);
26187         if(!this.overlay){
26188             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26189             o.unselectable();
26190             o.enableDisplayMode("block");
26191             // all splitbars share the same overlay
26192             Roo.SplitBar.prototype.overlay = o;
26193         }
26194         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26195         this.overlay.show();
26196         Roo.get(this.proxy).setDisplayed("block");
26197         var size = this.adapter.getElementSize(this);
26198         this.activeMinSize = this.getMinimumSize();;
26199         this.activeMaxSize = this.getMaximumSize();;
26200         var c1 = size - this.activeMinSize;
26201         var c2 = Math.max(this.activeMaxSize - size, 0);
26202         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26203             this.dd.resetConstraints();
26204             this.dd.setXConstraint(
26205                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26206                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26207             );
26208             this.dd.setYConstraint(0, 0);
26209         }else{
26210             this.dd.resetConstraints();
26211             this.dd.setXConstraint(0, 0);
26212             this.dd.setYConstraint(
26213                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26214                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26215             );
26216          }
26217         this.dragSpecs.startSize = size;
26218         this.dragSpecs.startPoint = [x, y];
26219         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26220     },
26221     
26222     /** 
26223      * @private Called after the drag operation by the DDProxy
26224      */
26225     onEndProxyDrag : function(e){
26226         Roo.get(this.proxy).setDisplayed(false);
26227         var endPoint = Roo.lib.Event.getXY(e);
26228         if(this.overlay){
26229             this.overlay.hide();
26230         }
26231         var newSize;
26232         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26233             newSize = this.dragSpecs.startSize + 
26234                 (this.placement == Roo.SplitBar.LEFT ?
26235                     endPoint[0] - this.dragSpecs.startPoint[0] :
26236                     this.dragSpecs.startPoint[0] - endPoint[0]
26237                 );
26238         }else{
26239             newSize = this.dragSpecs.startSize + 
26240                 (this.placement == Roo.SplitBar.TOP ?
26241                     endPoint[1] - this.dragSpecs.startPoint[1] :
26242                     this.dragSpecs.startPoint[1] - endPoint[1]
26243                 );
26244         }
26245         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26246         if(newSize != this.dragSpecs.startSize){
26247             if(this.fireEvent('beforeapply', this, newSize) !== false){
26248                 this.adapter.setElementSize(this, newSize);
26249                 this.fireEvent("moved", this, newSize);
26250                 this.fireEvent("resize", this, newSize);
26251             }
26252         }
26253     },
26254     
26255     /**
26256      * Get the adapter this SplitBar uses
26257      * @return The adapter object
26258      */
26259     getAdapter : function(){
26260         return this.adapter;
26261     },
26262     
26263     /**
26264      * Set the adapter this SplitBar uses
26265      * @param {Object} adapter A SplitBar adapter object
26266      */
26267     setAdapter : function(adapter){
26268         this.adapter = adapter;
26269         this.adapter.init(this);
26270     },
26271     
26272     /**
26273      * Gets the minimum size for the resizing element
26274      * @return {Number} The minimum size
26275      */
26276     getMinimumSize : function(){
26277         return this.minSize;
26278     },
26279     
26280     /**
26281      * Sets the minimum size for the resizing element
26282      * @param {Number} minSize The minimum size
26283      */
26284     setMinimumSize : function(minSize){
26285         this.minSize = minSize;
26286     },
26287     
26288     /**
26289      * Gets the maximum size for the resizing element
26290      * @return {Number} The maximum size
26291      */
26292     getMaximumSize : function(){
26293         return this.maxSize;
26294     },
26295     
26296     /**
26297      * Sets the maximum size for the resizing element
26298      * @param {Number} maxSize The maximum size
26299      */
26300     setMaximumSize : function(maxSize){
26301         this.maxSize = maxSize;
26302     },
26303     
26304     /**
26305      * Sets the initialize size for the resizing element
26306      * @param {Number} size The initial size
26307      */
26308     setCurrentSize : function(size){
26309         var oldAnimate = this.animate;
26310         this.animate = false;
26311         this.adapter.setElementSize(this, size);
26312         this.animate = oldAnimate;
26313     },
26314     
26315     /**
26316      * Destroy this splitbar. 
26317      * @param {Boolean} removeEl True to remove the element
26318      */
26319     destroy : function(removeEl){
26320         if(this.shim){
26321             this.shim.remove();
26322         }
26323         this.dd.unreg();
26324         this.proxy.parentNode.removeChild(this.proxy);
26325         if(removeEl){
26326             this.el.remove();
26327         }
26328     }
26329 });
26330
26331 /**
26332  * @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.
26333  */
26334 Roo.SplitBar.createProxy = function(dir){
26335     var proxy = new Roo.Element(document.createElement("div"));
26336     proxy.unselectable();
26337     var cls = 'x-splitbar-proxy';
26338     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26339     document.body.appendChild(proxy.dom);
26340     return proxy.dom;
26341 };
26342
26343 /** 
26344  * @class Roo.SplitBar.BasicLayoutAdapter
26345  * Default Adapter. It assumes the splitter and resizing element are not positioned
26346  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26347  */
26348 Roo.SplitBar.BasicLayoutAdapter = function(){
26349 };
26350
26351 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26352     // do nothing for now
26353     init : function(s){
26354     
26355     },
26356     /**
26357      * Called before drag operations to get the current size of the resizing element. 
26358      * @param {Roo.SplitBar} s The SplitBar using this adapter
26359      */
26360      getElementSize : function(s){
26361         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26362             return s.resizingEl.getWidth();
26363         }else{
26364             return s.resizingEl.getHeight();
26365         }
26366     },
26367     
26368     /**
26369      * Called after drag operations to set the size of the resizing element.
26370      * @param {Roo.SplitBar} s The SplitBar using this adapter
26371      * @param {Number} newSize The new size to set
26372      * @param {Function} onComplete A function to be invoked when resizing is complete
26373      */
26374     setElementSize : function(s, newSize, onComplete){
26375         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26376             if(!s.animate){
26377                 s.resizingEl.setWidth(newSize);
26378                 if(onComplete){
26379                     onComplete(s, newSize);
26380                 }
26381             }else{
26382                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26383             }
26384         }else{
26385             
26386             if(!s.animate){
26387                 s.resizingEl.setHeight(newSize);
26388                 if(onComplete){
26389                     onComplete(s, newSize);
26390                 }
26391             }else{
26392                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26393             }
26394         }
26395     }
26396 };
26397
26398 /** 
26399  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26400  * @extends Roo.SplitBar.BasicLayoutAdapter
26401  * Adapter that  moves the splitter element to align with the resized sizing element. 
26402  * Used with an absolute positioned SplitBar.
26403  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26404  * document.body, make sure you assign an id to the body element.
26405  */
26406 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26407     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26408     this.container = Roo.get(container);
26409 };
26410
26411 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26412     init : function(s){
26413         this.basic.init(s);
26414     },
26415     
26416     getElementSize : function(s){
26417         return this.basic.getElementSize(s);
26418     },
26419     
26420     setElementSize : function(s, newSize, onComplete){
26421         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26422     },
26423     
26424     moveSplitter : function(s){
26425         var yes = Roo.SplitBar;
26426         switch(s.placement){
26427             case yes.LEFT:
26428                 s.el.setX(s.resizingEl.getRight());
26429                 break;
26430             case yes.RIGHT:
26431                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26432                 break;
26433             case yes.TOP:
26434                 s.el.setY(s.resizingEl.getBottom());
26435                 break;
26436             case yes.BOTTOM:
26437                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26438                 break;
26439         }
26440     }
26441 };
26442
26443 /**
26444  * Orientation constant - Create a vertical SplitBar
26445  * @static
26446  * @type Number
26447  */
26448 Roo.SplitBar.VERTICAL = 1;
26449
26450 /**
26451  * Orientation constant - Create a horizontal SplitBar
26452  * @static
26453  * @type Number
26454  */
26455 Roo.SplitBar.HORIZONTAL = 2;
26456
26457 /**
26458  * Placement constant - The resizing element is to the left of the splitter element
26459  * @static
26460  * @type Number
26461  */
26462 Roo.SplitBar.LEFT = 1;
26463
26464 /**
26465  * Placement constant - The resizing element is to the right of the splitter element
26466  * @static
26467  * @type Number
26468  */
26469 Roo.SplitBar.RIGHT = 2;
26470
26471 /**
26472  * Placement constant - The resizing element is positioned above the splitter element
26473  * @static
26474  * @type Number
26475  */
26476 Roo.SplitBar.TOP = 3;
26477
26478 /**
26479  * Placement constant - The resizing element is positioned under splitter element
26480  * @static
26481  * @type Number
26482  */
26483 Roo.SplitBar.BOTTOM = 4;
26484 /*
26485  * Based on:
26486  * Ext JS Library 1.1.1
26487  * Copyright(c) 2006-2007, Ext JS, LLC.
26488  *
26489  * Originally Released Under LGPL - original licence link has changed is not relivant.
26490  *
26491  * Fork - LGPL
26492  * <script type="text/javascript">
26493  */
26494
26495 /**
26496  * @class Roo.View
26497  * @extends Roo.util.Observable
26498  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26499  * This class also supports single and multi selection modes. <br>
26500  * Create a data model bound view:
26501  <pre><code>
26502  var store = new Roo.data.Store(...);
26503
26504  var view = new Roo.View({
26505     el : "my-element",
26506     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26507  
26508     singleSelect: true,
26509     selectedClass: "ydataview-selected",
26510     store: store
26511  });
26512
26513  // listen for node click?
26514  view.on("click", function(vw, index, node, e){
26515  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26516  });
26517
26518  // load XML data
26519  dataModel.load("foobar.xml");
26520  </code></pre>
26521  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26522  * <br><br>
26523  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26524  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26525  * 
26526  * Note: old style constructor is still suported (container, template, config)
26527  * 
26528  * @constructor
26529  * Create a new View
26530  * @param {Object} config The config object
26531  * 
26532  */
26533 Roo.View = function(config, depreciated_tpl, depreciated_config){
26534     
26535     this.parent = false;
26536     
26537     if (typeof(depreciated_tpl) == 'undefined') {
26538         // new way.. - universal constructor.
26539         Roo.apply(this, config);
26540         this.el  = Roo.get(this.el);
26541     } else {
26542         // old format..
26543         this.el  = Roo.get(config);
26544         this.tpl = depreciated_tpl;
26545         Roo.apply(this, depreciated_config);
26546     }
26547     this.wrapEl  = this.el.wrap().wrap();
26548     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26549     
26550     
26551     if(typeof(this.tpl) == "string"){
26552         this.tpl = new Roo.Template(this.tpl);
26553     } else {
26554         // support xtype ctors..
26555         this.tpl = new Roo.factory(this.tpl, Roo);
26556     }
26557     
26558     
26559     this.tpl.compile();
26560     
26561     /** @private */
26562     this.addEvents({
26563         /**
26564          * @event beforeclick
26565          * Fires before a click is processed. Returns false to cancel the default action.
26566          * @param {Roo.View} this
26567          * @param {Number} index The index of the target node
26568          * @param {HTMLElement} node The target node
26569          * @param {Roo.EventObject} e The raw event object
26570          */
26571             "beforeclick" : true,
26572         /**
26573          * @event click
26574          * Fires when a template node is clicked.
26575          * @param {Roo.View} this
26576          * @param {Number} index The index of the target node
26577          * @param {HTMLElement} node The target node
26578          * @param {Roo.EventObject} e The raw event object
26579          */
26580             "click" : true,
26581         /**
26582          * @event dblclick
26583          * Fires when a template node is double clicked.
26584          * @param {Roo.View} this
26585          * @param {Number} index The index of the target node
26586          * @param {HTMLElement} node The target node
26587          * @param {Roo.EventObject} e The raw event object
26588          */
26589             "dblclick" : true,
26590         /**
26591          * @event contextmenu
26592          * Fires when a template node is right clicked.
26593          * @param {Roo.View} this
26594          * @param {Number} index The index of the target node
26595          * @param {HTMLElement} node The target node
26596          * @param {Roo.EventObject} e The raw event object
26597          */
26598             "contextmenu" : true,
26599         /**
26600          * @event selectionchange
26601          * Fires when the selected nodes change.
26602          * @param {Roo.View} this
26603          * @param {Array} selections Array of the selected nodes
26604          */
26605             "selectionchange" : true,
26606     
26607         /**
26608          * @event beforeselect
26609          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26610          * @param {Roo.View} this
26611          * @param {HTMLElement} node The node to be selected
26612          * @param {Array} selections Array of currently selected nodes
26613          */
26614             "beforeselect" : true,
26615         /**
26616          * @event preparedata
26617          * Fires on every row to render, to allow you to change the data.
26618          * @param {Roo.View} this
26619          * @param {Object} data to be rendered (change this)
26620          */
26621           "preparedata" : true
26622           
26623           
26624         });
26625
26626
26627
26628     this.el.on({
26629         "click": this.onClick,
26630         "dblclick": this.onDblClick,
26631         "contextmenu": this.onContextMenu,
26632         scope:this
26633     });
26634
26635     this.selections = [];
26636     this.nodes = [];
26637     this.cmp = new Roo.CompositeElementLite([]);
26638     if(this.store){
26639         this.store = Roo.factory(this.store, Roo.data);
26640         this.setStore(this.store, true);
26641     }
26642     
26643     if ( this.footer && this.footer.xtype) {
26644            
26645          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26646         
26647         this.footer.dataSource = this.store;
26648         this.footer.container = fctr;
26649         this.footer = Roo.factory(this.footer, Roo);
26650         fctr.insertFirst(this.el);
26651         
26652         // this is a bit insane - as the paging toolbar seems to detach the el..
26653 //        dom.parentNode.parentNode.parentNode
26654          // they get detached?
26655     }
26656     
26657     
26658     Roo.View.superclass.constructor.call(this);
26659     
26660     
26661 };
26662
26663 Roo.extend(Roo.View, Roo.util.Observable, {
26664     
26665      /**
26666      * @cfg {Roo.data.Store} store Data store to load data from.
26667      */
26668     store : false,
26669     
26670     /**
26671      * @cfg {String|Roo.Element} el The container element.
26672      */
26673     el : '',
26674     
26675     /**
26676      * @cfg {String|Roo.Template} tpl The template used by this View 
26677      */
26678     tpl : false,
26679     /**
26680      * @cfg {String} dataName the named area of the template to use as the data area
26681      *                          Works with domtemplates roo-name="name"
26682      */
26683     dataName: false,
26684     /**
26685      * @cfg {String} selectedClass The css class to add to selected nodes
26686      */
26687     selectedClass : "x-view-selected",
26688      /**
26689      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26690      */
26691     emptyText : "",
26692     
26693     /**
26694      * @cfg {String} text to display on mask (default Loading)
26695      */
26696     mask : false,
26697     /**
26698      * @cfg {Boolean} multiSelect Allow multiple selection
26699      */
26700     multiSelect : false,
26701     /**
26702      * @cfg {Boolean} singleSelect Allow single selection
26703      */
26704     singleSelect:  false,
26705     
26706     /**
26707      * @cfg {Boolean} toggleSelect - selecting 
26708      */
26709     toggleSelect : false,
26710     
26711     /**
26712      * @cfg {Boolean} tickable - selecting 
26713      */
26714     tickable : false,
26715     
26716     /**
26717      * Returns the element this view is bound to.
26718      * @return {Roo.Element}
26719      */
26720     getEl : function(){
26721         return this.wrapEl;
26722     },
26723     
26724     
26725
26726     /**
26727      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26728      */
26729     refresh : function(){
26730         //Roo.log('refresh');
26731         var t = this.tpl;
26732         
26733         // if we are using something like 'domtemplate', then
26734         // the what gets used is:
26735         // t.applySubtemplate(NAME, data, wrapping data..)
26736         // the outer template then get' applied with
26737         //     the store 'extra data'
26738         // and the body get's added to the
26739         //      roo-name="data" node?
26740         //      <span class='roo-tpl-{name}'></span> ?????
26741         
26742         
26743         
26744         this.clearSelections();
26745         this.el.update("");
26746         var html = [];
26747         var records = this.store.getRange();
26748         if(records.length < 1) {
26749             
26750             // is this valid??  = should it render a template??
26751             
26752             this.el.update(this.emptyText);
26753             return;
26754         }
26755         var el = this.el;
26756         if (this.dataName) {
26757             this.el.update(t.apply(this.store.meta)); //????
26758             el = this.el.child('.roo-tpl-' + this.dataName);
26759         }
26760         
26761         for(var i = 0, len = records.length; i < len; i++){
26762             var data = this.prepareData(records[i].data, i, records[i]);
26763             this.fireEvent("preparedata", this, data, i, records[i]);
26764             
26765             var d = Roo.apply({}, data);
26766             
26767             if(this.tickable){
26768                 Roo.apply(d, {'roo-id' : Roo.id()});
26769                 
26770                 var _this = this;
26771             
26772                 Roo.each(this.parent.item, function(item){
26773                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26774                         return;
26775                     }
26776                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26777                 });
26778             }
26779             
26780             html[html.length] = Roo.util.Format.trim(
26781                 this.dataName ?
26782                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26783                     t.apply(d)
26784             );
26785         }
26786         
26787         
26788         
26789         el.update(html.join(""));
26790         this.nodes = el.dom.childNodes;
26791         this.updateIndexes(0);
26792     },
26793     
26794
26795     /**
26796      * Function to override to reformat the data that is sent to
26797      * the template for each node.
26798      * DEPRICATED - use the preparedata event handler.
26799      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26800      * a JSON object for an UpdateManager bound view).
26801      */
26802     prepareData : function(data, index, record)
26803     {
26804         this.fireEvent("preparedata", this, data, index, record);
26805         return data;
26806     },
26807
26808     onUpdate : function(ds, record){
26809         // Roo.log('on update');   
26810         this.clearSelections();
26811         var index = this.store.indexOf(record);
26812         var n = this.nodes[index];
26813         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26814         n.parentNode.removeChild(n);
26815         this.updateIndexes(index, index);
26816     },
26817
26818     
26819     
26820 // --------- FIXME     
26821     onAdd : function(ds, records, index)
26822     {
26823         //Roo.log(['on Add', ds, records, index] );        
26824         this.clearSelections();
26825         if(this.nodes.length == 0){
26826             this.refresh();
26827             return;
26828         }
26829         var n = this.nodes[index];
26830         for(var i = 0, len = records.length; i < len; i++){
26831             var d = this.prepareData(records[i].data, i, records[i]);
26832             if(n){
26833                 this.tpl.insertBefore(n, d);
26834             }else{
26835                 
26836                 this.tpl.append(this.el, d);
26837             }
26838         }
26839         this.updateIndexes(index);
26840     },
26841
26842     onRemove : function(ds, record, index){
26843        // Roo.log('onRemove');
26844         this.clearSelections();
26845         var el = this.dataName  ?
26846             this.el.child('.roo-tpl-' + this.dataName) :
26847             this.el; 
26848         
26849         el.dom.removeChild(this.nodes[index]);
26850         this.updateIndexes(index);
26851     },
26852
26853     /**
26854      * Refresh an individual node.
26855      * @param {Number} index
26856      */
26857     refreshNode : function(index){
26858         this.onUpdate(this.store, this.store.getAt(index));
26859     },
26860
26861     updateIndexes : function(startIndex, endIndex){
26862         var ns = this.nodes;
26863         startIndex = startIndex || 0;
26864         endIndex = endIndex || ns.length - 1;
26865         for(var i = startIndex; i <= endIndex; i++){
26866             ns[i].nodeIndex = i;
26867         }
26868     },
26869
26870     /**
26871      * Changes the data store this view uses and refresh the view.
26872      * @param {Store} store
26873      */
26874     setStore : function(store, initial){
26875         if(!initial && this.store){
26876             this.store.un("datachanged", this.refresh);
26877             this.store.un("add", this.onAdd);
26878             this.store.un("remove", this.onRemove);
26879             this.store.un("update", this.onUpdate);
26880             this.store.un("clear", this.refresh);
26881             this.store.un("beforeload", this.onBeforeLoad);
26882             this.store.un("load", this.onLoad);
26883             this.store.un("loadexception", this.onLoad);
26884         }
26885         if(store){
26886           
26887             store.on("datachanged", this.refresh, this);
26888             store.on("add", this.onAdd, this);
26889             store.on("remove", this.onRemove, this);
26890             store.on("update", this.onUpdate, this);
26891             store.on("clear", this.refresh, this);
26892             store.on("beforeload", this.onBeforeLoad, this);
26893             store.on("load", this.onLoad, this);
26894             store.on("loadexception", this.onLoad, this);
26895         }
26896         
26897         if(store){
26898             this.refresh();
26899         }
26900     },
26901     /**
26902      * onbeforeLoad - masks the loading area.
26903      *
26904      */
26905     onBeforeLoad : function(store,opts)
26906     {
26907          //Roo.log('onBeforeLoad');   
26908         if (!opts.add) {
26909             this.el.update("");
26910         }
26911         this.el.mask(this.mask ? this.mask : "Loading" ); 
26912     },
26913     onLoad : function ()
26914     {
26915         this.el.unmask();
26916     },
26917     
26918
26919     /**
26920      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26921      * @param {HTMLElement} node
26922      * @return {HTMLElement} The template node
26923      */
26924     findItemFromChild : function(node){
26925         var el = this.dataName  ?
26926             this.el.child('.roo-tpl-' + this.dataName,true) :
26927             this.el.dom; 
26928         
26929         if(!node || node.parentNode == el){
26930                     return node;
26931             }
26932             var p = node.parentNode;
26933             while(p && p != el){
26934             if(p.parentNode == el){
26935                 return p;
26936             }
26937             p = p.parentNode;
26938         }
26939             return null;
26940     },
26941
26942     /** @ignore */
26943     onClick : function(e){
26944         var item = this.findItemFromChild(e.getTarget());
26945         if(item){
26946             var index = this.indexOf(item);
26947             if(this.onItemClick(item, index, e) !== false){
26948                 this.fireEvent("click", this, index, item, e);
26949             }
26950         }else{
26951             this.clearSelections();
26952         }
26953     },
26954
26955     /** @ignore */
26956     onContextMenu : function(e){
26957         var item = this.findItemFromChild(e.getTarget());
26958         if(item){
26959             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26960         }
26961     },
26962
26963     /** @ignore */
26964     onDblClick : function(e){
26965         var item = this.findItemFromChild(e.getTarget());
26966         if(item){
26967             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
26968         }
26969     },
26970
26971     onItemClick : function(item, index, e)
26972     {
26973         if(this.fireEvent("beforeclick", this, index, item, e) === false){
26974             return false;
26975         }
26976         if (this.toggleSelect) {
26977             var m = this.isSelected(item) ? 'unselect' : 'select';
26978             //Roo.log(m);
26979             var _t = this;
26980             _t[m](item, true, false);
26981             return true;
26982         }
26983         if(this.multiSelect || this.singleSelect){
26984             if(this.multiSelect && e.shiftKey && this.lastSelection){
26985                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
26986             }else{
26987                 this.select(item, this.multiSelect && e.ctrlKey);
26988                 this.lastSelection = item;
26989             }
26990             
26991             if(!this.tickable){
26992                 e.preventDefault();
26993             }
26994             
26995         }
26996         return true;
26997     },
26998
26999     /**
27000      * Get the number of selected nodes.
27001      * @return {Number}
27002      */
27003     getSelectionCount : function(){
27004         return this.selections.length;
27005     },
27006
27007     /**
27008      * Get the currently selected nodes.
27009      * @return {Array} An array of HTMLElements
27010      */
27011     getSelectedNodes : function(){
27012         return this.selections;
27013     },
27014
27015     /**
27016      * Get the indexes of the selected nodes.
27017      * @return {Array}
27018      */
27019     getSelectedIndexes : function(){
27020         var indexes = [], s = this.selections;
27021         for(var i = 0, len = s.length; i < len; i++){
27022             indexes.push(s[i].nodeIndex);
27023         }
27024         return indexes;
27025     },
27026
27027     /**
27028      * Clear all selections
27029      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27030      */
27031     clearSelections : function(suppressEvent){
27032         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27033             this.cmp.elements = this.selections;
27034             this.cmp.removeClass(this.selectedClass);
27035             this.selections = [];
27036             if(!suppressEvent){
27037                 this.fireEvent("selectionchange", this, this.selections);
27038             }
27039         }
27040     },
27041
27042     /**
27043      * Returns true if the passed node is selected
27044      * @param {HTMLElement/Number} node The node or node index
27045      * @return {Boolean}
27046      */
27047     isSelected : function(node){
27048         var s = this.selections;
27049         if(s.length < 1){
27050             return false;
27051         }
27052         node = this.getNode(node);
27053         return s.indexOf(node) !== -1;
27054     },
27055
27056     /**
27057      * Selects nodes.
27058      * @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
27059      * @param {Boolean} keepExisting (optional) true to keep existing selections
27060      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27061      */
27062     select : function(nodeInfo, keepExisting, suppressEvent){
27063         if(nodeInfo instanceof Array){
27064             if(!keepExisting){
27065                 this.clearSelections(true);
27066             }
27067             for(var i = 0, len = nodeInfo.length; i < len; i++){
27068                 this.select(nodeInfo[i], true, true);
27069             }
27070             return;
27071         } 
27072         var node = this.getNode(nodeInfo);
27073         if(!node || this.isSelected(node)){
27074             return; // already selected.
27075         }
27076         if(!keepExisting){
27077             this.clearSelections(true);
27078         }
27079         
27080         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27081             Roo.fly(node).addClass(this.selectedClass);
27082             this.selections.push(node);
27083             if(!suppressEvent){
27084                 this.fireEvent("selectionchange", this, this.selections);
27085             }
27086         }
27087         
27088         
27089     },
27090       /**
27091      * Unselects nodes.
27092      * @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
27093      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27094      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27095      */
27096     unselect : function(nodeInfo, keepExisting, suppressEvent)
27097     {
27098         if(nodeInfo instanceof Array){
27099             Roo.each(this.selections, function(s) {
27100                 this.unselect(s, nodeInfo);
27101             }, this);
27102             return;
27103         }
27104         var node = this.getNode(nodeInfo);
27105         if(!node || !this.isSelected(node)){
27106             //Roo.log("not selected");
27107             return; // not selected.
27108         }
27109         // fireevent???
27110         var ns = [];
27111         Roo.each(this.selections, function(s) {
27112             if (s == node ) {
27113                 Roo.fly(node).removeClass(this.selectedClass);
27114
27115                 return;
27116             }
27117             ns.push(s);
27118         },this);
27119         
27120         this.selections= ns;
27121         this.fireEvent("selectionchange", this, this.selections);
27122     },
27123
27124     /**
27125      * Gets a template node.
27126      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27127      * @return {HTMLElement} The node or null if it wasn't found
27128      */
27129     getNode : function(nodeInfo){
27130         if(typeof nodeInfo == "string"){
27131             return document.getElementById(nodeInfo);
27132         }else if(typeof nodeInfo == "number"){
27133             return this.nodes[nodeInfo];
27134         }
27135         return nodeInfo;
27136     },
27137
27138     /**
27139      * Gets a range template nodes.
27140      * @param {Number} startIndex
27141      * @param {Number} endIndex
27142      * @return {Array} An array of nodes
27143      */
27144     getNodes : function(start, end){
27145         var ns = this.nodes;
27146         start = start || 0;
27147         end = typeof end == "undefined" ? ns.length - 1 : end;
27148         var nodes = [];
27149         if(start <= end){
27150             for(var i = start; i <= end; i++){
27151                 nodes.push(ns[i]);
27152             }
27153         } else{
27154             for(var i = start; i >= end; i--){
27155                 nodes.push(ns[i]);
27156             }
27157         }
27158         return nodes;
27159     },
27160
27161     /**
27162      * Finds the index of the passed node
27163      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27164      * @return {Number} The index of the node or -1
27165      */
27166     indexOf : function(node){
27167         node = this.getNode(node);
27168         if(typeof node.nodeIndex == "number"){
27169             return node.nodeIndex;
27170         }
27171         var ns = this.nodes;
27172         for(var i = 0, len = ns.length; i < len; i++){
27173             if(ns[i] == node){
27174                 return i;
27175             }
27176         }
27177         return -1;
27178     }
27179 });
27180 /*
27181  * Based on:
27182  * Ext JS Library 1.1.1
27183  * Copyright(c) 2006-2007, Ext JS, LLC.
27184  *
27185  * Originally Released Under LGPL - original licence link has changed is not relivant.
27186  *
27187  * Fork - LGPL
27188  * <script type="text/javascript">
27189  */
27190
27191 /**
27192  * @class Roo.JsonView
27193  * @extends Roo.View
27194  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27195 <pre><code>
27196 var view = new Roo.JsonView({
27197     container: "my-element",
27198     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27199     multiSelect: true, 
27200     jsonRoot: "data" 
27201 });
27202
27203 // listen for node click?
27204 view.on("click", function(vw, index, node, e){
27205     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27206 });
27207
27208 // direct load of JSON data
27209 view.load("foobar.php");
27210
27211 // Example from my blog list
27212 var tpl = new Roo.Template(
27213     '&lt;div class="entry"&gt;' +
27214     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27215     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27216     "&lt;/div&gt;&lt;hr /&gt;"
27217 );
27218
27219 var moreView = new Roo.JsonView({
27220     container :  "entry-list", 
27221     template : tpl,
27222     jsonRoot: "posts"
27223 });
27224 moreView.on("beforerender", this.sortEntries, this);
27225 moreView.load({
27226     url: "/blog/get-posts.php",
27227     params: "allposts=true",
27228     text: "Loading Blog Entries..."
27229 });
27230 </code></pre>
27231
27232 * Note: old code is supported with arguments : (container, template, config)
27233
27234
27235  * @constructor
27236  * Create a new JsonView
27237  * 
27238  * @param {Object} config The config object
27239  * 
27240  */
27241 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27242     
27243     
27244     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27245
27246     var um = this.el.getUpdateManager();
27247     um.setRenderer(this);
27248     um.on("update", this.onLoad, this);
27249     um.on("failure", this.onLoadException, this);
27250
27251     /**
27252      * @event beforerender
27253      * Fires before rendering of the downloaded JSON data.
27254      * @param {Roo.JsonView} this
27255      * @param {Object} data The JSON data loaded
27256      */
27257     /**
27258      * @event load
27259      * Fires when data is loaded.
27260      * @param {Roo.JsonView} this
27261      * @param {Object} data The JSON data loaded
27262      * @param {Object} response The raw Connect response object
27263      */
27264     /**
27265      * @event loadexception
27266      * Fires when loading fails.
27267      * @param {Roo.JsonView} this
27268      * @param {Object} response The raw Connect response object
27269      */
27270     this.addEvents({
27271         'beforerender' : true,
27272         'load' : true,
27273         'loadexception' : true
27274     });
27275 };
27276 Roo.extend(Roo.JsonView, Roo.View, {
27277     /**
27278      * @type {String} The root property in the loaded JSON object that contains the data
27279      */
27280     jsonRoot : "",
27281
27282     /**
27283      * Refreshes the view.
27284      */
27285     refresh : function(){
27286         this.clearSelections();
27287         this.el.update("");
27288         var html = [];
27289         var o = this.jsonData;
27290         if(o && o.length > 0){
27291             for(var i = 0, len = o.length; i < len; i++){
27292                 var data = this.prepareData(o[i], i, o);
27293                 html[html.length] = this.tpl.apply(data);
27294             }
27295         }else{
27296             html.push(this.emptyText);
27297         }
27298         this.el.update(html.join(""));
27299         this.nodes = this.el.dom.childNodes;
27300         this.updateIndexes(0);
27301     },
27302
27303     /**
27304      * 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.
27305      * @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:
27306      <pre><code>
27307      view.load({
27308          url: "your-url.php",
27309          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27310          callback: yourFunction,
27311          scope: yourObject, //(optional scope)
27312          discardUrl: false,
27313          nocache: false,
27314          text: "Loading...",
27315          timeout: 30,
27316          scripts: false
27317      });
27318      </code></pre>
27319      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27320      * 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.
27321      * @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}
27322      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27323      * @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.
27324      */
27325     load : function(){
27326         var um = this.el.getUpdateManager();
27327         um.update.apply(um, arguments);
27328     },
27329
27330     // note - render is a standard framework call...
27331     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27332     render : function(el, response){
27333         
27334         this.clearSelections();
27335         this.el.update("");
27336         var o;
27337         try{
27338             if (response != '') {
27339                 o = Roo.util.JSON.decode(response.responseText);
27340                 if(this.jsonRoot){
27341                     
27342                     o = o[this.jsonRoot];
27343                 }
27344             }
27345         } catch(e){
27346         }
27347         /**
27348          * The current JSON data or null
27349          */
27350         this.jsonData = o;
27351         this.beforeRender();
27352         this.refresh();
27353     },
27354
27355 /**
27356  * Get the number of records in the current JSON dataset
27357  * @return {Number}
27358  */
27359     getCount : function(){
27360         return this.jsonData ? this.jsonData.length : 0;
27361     },
27362
27363 /**
27364  * Returns the JSON object for the specified node(s)
27365  * @param {HTMLElement/Array} node The node or an array of nodes
27366  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27367  * you get the JSON object for the node
27368  */
27369     getNodeData : function(node){
27370         if(node instanceof Array){
27371             var data = [];
27372             for(var i = 0, len = node.length; i < len; i++){
27373                 data.push(this.getNodeData(node[i]));
27374             }
27375             return data;
27376         }
27377         return this.jsonData[this.indexOf(node)] || null;
27378     },
27379
27380     beforeRender : function(){
27381         this.snapshot = this.jsonData;
27382         if(this.sortInfo){
27383             this.sort.apply(this, this.sortInfo);
27384         }
27385         this.fireEvent("beforerender", this, this.jsonData);
27386     },
27387
27388     onLoad : function(el, o){
27389         this.fireEvent("load", this, this.jsonData, o);
27390     },
27391
27392     onLoadException : function(el, o){
27393         this.fireEvent("loadexception", this, o);
27394     },
27395
27396 /**
27397  * Filter the data by a specific property.
27398  * @param {String} property A property on your JSON objects
27399  * @param {String/RegExp} value Either string that the property values
27400  * should start with, or a RegExp to test against the property
27401  */
27402     filter : function(property, value){
27403         if(this.jsonData){
27404             var data = [];
27405             var ss = this.snapshot;
27406             if(typeof value == "string"){
27407                 var vlen = value.length;
27408                 if(vlen == 0){
27409                     this.clearFilter();
27410                     return;
27411                 }
27412                 value = value.toLowerCase();
27413                 for(var i = 0, len = ss.length; i < len; i++){
27414                     var o = ss[i];
27415                     if(o[property].substr(0, vlen).toLowerCase() == value){
27416                         data.push(o);
27417                     }
27418                 }
27419             } else if(value.exec){ // regex?
27420                 for(var i = 0, len = ss.length; i < len; i++){
27421                     var o = ss[i];
27422                     if(value.test(o[property])){
27423                         data.push(o);
27424                     }
27425                 }
27426             } else{
27427                 return;
27428             }
27429             this.jsonData = data;
27430             this.refresh();
27431         }
27432     },
27433
27434 /**
27435  * Filter by a function. The passed function will be called with each
27436  * object in the current dataset. If the function returns true the value is kept,
27437  * otherwise it is filtered.
27438  * @param {Function} fn
27439  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27440  */
27441     filterBy : function(fn, scope){
27442         if(this.jsonData){
27443             var data = [];
27444             var ss = this.snapshot;
27445             for(var i = 0, len = ss.length; i < len; i++){
27446                 var o = ss[i];
27447                 if(fn.call(scope || this, o)){
27448                     data.push(o);
27449                 }
27450             }
27451             this.jsonData = data;
27452             this.refresh();
27453         }
27454     },
27455
27456 /**
27457  * Clears the current filter.
27458  */
27459     clearFilter : function(){
27460         if(this.snapshot && this.jsonData != this.snapshot){
27461             this.jsonData = this.snapshot;
27462             this.refresh();
27463         }
27464     },
27465
27466
27467 /**
27468  * Sorts the data for this view and refreshes it.
27469  * @param {String} property A property on your JSON objects to sort on
27470  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27471  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27472  */
27473     sort : function(property, dir, sortType){
27474         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27475         if(this.jsonData){
27476             var p = property;
27477             var dsc = dir && dir.toLowerCase() == "desc";
27478             var f = function(o1, o2){
27479                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27480                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27481                 ;
27482                 if(v1 < v2){
27483                     return dsc ? +1 : -1;
27484                 } else if(v1 > v2){
27485                     return dsc ? -1 : +1;
27486                 } else{
27487                     return 0;
27488                 }
27489             };
27490             this.jsonData.sort(f);
27491             this.refresh();
27492             if(this.jsonData != this.snapshot){
27493                 this.snapshot.sort(f);
27494             }
27495         }
27496     }
27497 });/*
27498  * Based on:
27499  * Ext JS Library 1.1.1
27500  * Copyright(c) 2006-2007, Ext JS, LLC.
27501  *
27502  * Originally Released Under LGPL - original licence link has changed is not relivant.
27503  *
27504  * Fork - LGPL
27505  * <script type="text/javascript">
27506  */
27507  
27508
27509 /**
27510  * @class Roo.ColorPalette
27511  * @extends Roo.Component
27512  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27513  * Here's an example of typical usage:
27514  * <pre><code>
27515 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27516 cp.render('my-div');
27517
27518 cp.on('select', function(palette, selColor){
27519     // do something with selColor
27520 });
27521 </code></pre>
27522  * @constructor
27523  * Create a new ColorPalette
27524  * @param {Object} config The config object
27525  */
27526 Roo.ColorPalette = function(config){
27527     Roo.ColorPalette.superclass.constructor.call(this, config);
27528     this.addEvents({
27529         /**
27530              * @event select
27531              * Fires when a color is selected
27532              * @param {ColorPalette} this
27533              * @param {String} color The 6-digit color hex code (without the # symbol)
27534              */
27535         select: true
27536     });
27537
27538     if(this.handler){
27539         this.on("select", this.handler, this.scope, true);
27540     }
27541 };
27542 Roo.extend(Roo.ColorPalette, Roo.Component, {
27543     /**
27544      * @cfg {String} itemCls
27545      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27546      */
27547     itemCls : "x-color-palette",
27548     /**
27549      * @cfg {String} value
27550      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27551      * the hex codes are case-sensitive.
27552      */
27553     value : null,
27554     clickEvent:'click',
27555     // private
27556     ctype: "Roo.ColorPalette",
27557
27558     /**
27559      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27560      */
27561     allowReselect : false,
27562
27563     /**
27564      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27565      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27566      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27567      * of colors with the width setting until the box is symmetrical.</p>
27568      * <p>You can override individual colors if needed:</p>
27569      * <pre><code>
27570 var cp = new Roo.ColorPalette();
27571 cp.colors[0] = "FF0000";  // change the first box to red
27572 </code></pre>
27573
27574 Or you can provide a custom array of your own for complete control:
27575 <pre><code>
27576 var cp = new Roo.ColorPalette();
27577 cp.colors = ["000000", "993300", "333300"];
27578 </code></pre>
27579      * @type Array
27580      */
27581     colors : [
27582         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27583         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27584         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27585         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27586         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27587     ],
27588
27589     // private
27590     onRender : function(container, position){
27591         var t = new Roo.MasterTemplate(
27592             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27593         );
27594         var c = this.colors;
27595         for(var i = 0, len = c.length; i < len; i++){
27596             t.add([c[i]]);
27597         }
27598         var el = document.createElement("div");
27599         el.className = this.itemCls;
27600         t.overwrite(el);
27601         container.dom.insertBefore(el, position);
27602         this.el = Roo.get(el);
27603         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27604         if(this.clickEvent != 'click'){
27605             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27606         }
27607     },
27608
27609     // private
27610     afterRender : function(){
27611         Roo.ColorPalette.superclass.afterRender.call(this);
27612         if(this.value){
27613             var s = this.value;
27614             this.value = null;
27615             this.select(s);
27616         }
27617     },
27618
27619     // private
27620     handleClick : function(e, t){
27621         e.preventDefault();
27622         if(!this.disabled){
27623             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27624             this.select(c.toUpperCase());
27625         }
27626     },
27627
27628     /**
27629      * Selects the specified color in the palette (fires the select event)
27630      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27631      */
27632     select : function(color){
27633         color = color.replace("#", "");
27634         if(color != this.value || this.allowReselect){
27635             var el = this.el;
27636             if(this.value){
27637                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27638             }
27639             el.child("a.color-"+color).addClass("x-color-palette-sel");
27640             this.value = color;
27641             this.fireEvent("select", this, color);
27642         }
27643     }
27644 });/*
27645  * Based on:
27646  * Ext JS Library 1.1.1
27647  * Copyright(c) 2006-2007, Ext JS, LLC.
27648  *
27649  * Originally Released Under LGPL - original licence link has changed is not relivant.
27650  *
27651  * Fork - LGPL
27652  * <script type="text/javascript">
27653  */
27654  
27655 /**
27656  * @class Roo.DatePicker
27657  * @extends Roo.Component
27658  * Simple date picker class.
27659  * @constructor
27660  * Create a new DatePicker
27661  * @param {Object} config The config object
27662  */
27663 Roo.DatePicker = function(config){
27664     Roo.DatePicker.superclass.constructor.call(this, config);
27665
27666     this.value = config && config.value ?
27667                  config.value.clearTime() : new Date().clearTime();
27668
27669     this.addEvents({
27670         /**
27671              * @event select
27672              * Fires when a date is selected
27673              * @param {DatePicker} this
27674              * @param {Date} date The selected date
27675              */
27676         'select': true,
27677         /**
27678              * @event monthchange
27679              * Fires when the displayed month changes 
27680              * @param {DatePicker} this
27681              * @param {Date} date The selected month
27682              */
27683         'monthchange': true
27684     });
27685
27686     if(this.handler){
27687         this.on("select", this.handler,  this.scope || this);
27688     }
27689     // build the disabledDatesRE
27690     if(!this.disabledDatesRE && this.disabledDates){
27691         var dd = this.disabledDates;
27692         var re = "(?:";
27693         for(var i = 0; i < dd.length; i++){
27694             re += dd[i];
27695             if(i != dd.length-1) {
27696                 re += "|";
27697             }
27698         }
27699         this.disabledDatesRE = new RegExp(re + ")");
27700     }
27701 };
27702
27703 Roo.extend(Roo.DatePicker, Roo.Component, {
27704     /**
27705      * @cfg {String} todayText
27706      * The text to display on the button that selects the current date (defaults to "Today")
27707      */
27708     todayText : "Today",
27709     /**
27710      * @cfg {String} okText
27711      * The text to display on the ok button
27712      */
27713     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27714     /**
27715      * @cfg {String} cancelText
27716      * The text to display on the cancel button
27717      */
27718     cancelText : "Cancel",
27719     /**
27720      * @cfg {String} todayTip
27721      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27722      */
27723     todayTip : "{0} (Spacebar)",
27724     /**
27725      * @cfg {Date} minDate
27726      * Minimum allowable date (JavaScript date object, defaults to null)
27727      */
27728     minDate : null,
27729     /**
27730      * @cfg {Date} maxDate
27731      * Maximum allowable date (JavaScript date object, defaults to null)
27732      */
27733     maxDate : null,
27734     /**
27735      * @cfg {String} minText
27736      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27737      */
27738     minText : "This date is before the minimum date",
27739     /**
27740      * @cfg {String} maxText
27741      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27742      */
27743     maxText : "This date is after the maximum date",
27744     /**
27745      * @cfg {String} format
27746      * The default date format string which can be overriden for localization support.  The format must be
27747      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27748      */
27749     format : "m/d/y",
27750     /**
27751      * @cfg {Array} disabledDays
27752      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27753      */
27754     disabledDays : null,
27755     /**
27756      * @cfg {String} disabledDaysText
27757      * The tooltip to display when the date falls on a disabled day (defaults to "")
27758      */
27759     disabledDaysText : "",
27760     /**
27761      * @cfg {RegExp} disabledDatesRE
27762      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27763      */
27764     disabledDatesRE : null,
27765     /**
27766      * @cfg {String} disabledDatesText
27767      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27768      */
27769     disabledDatesText : "",
27770     /**
27771      * @cfg {Boolean} constrainToViewport
27772      * True to constrain the date picker to the viewport (defaults to true)
27773      */
27774     constrainToViewport : true,
27775     /**
27776      * @cfg {Array} monthNames
27777      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27778      */
27779     monthNames : Date.monthNames,
27780     /**
27781      * @cfg {Array} dayNames
27782      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27783      */
27784     dayNames : Date.dayNames,
27785     /**
27786      * @cfg {String} nextText
27787      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27788      */
27789     nextText: 'Next Month (Control+Right)',
27790     /**
27791      * @cfg {String} prevText
27792      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27793      */
27794     prevText: 'Previous Month (Control+Left)',
27795     /**
27796      * @cfg {String} monthYearText
27797      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27798      */
27799     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27800     /**
27801      * @cfg {Number} startDay
27802      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27803      */
27804     startDay : 0,
27805     /**
27806      * @cfg {Bool} showClear
27807      * Show a clear button (usefull for date form elements that can be blank.)
27808      */
27809     
27810     showClear: false,
27811     
27812     /**
27813      * Sets the value of the date field
27814      * @param {Date} value The date to set
27815      */
27816     setValue : function(value){
27817         var old = this.value;
27818         
27819         if (typeof(value) == 'string') {
27820          
27821             value = Date.parseDate(value, this.format);
27822         }
27823         if (!value) {
27824             value = new Date();
27825         }
27826         
27827         this.value = value.clearTime(true);
27828         if(this.el){
27829             this.update(this.value);
27830         }
27831     },
27832
27833     /**
27834      * Gets the current selected value of the date field
27835      * @return {Date} The selected date
27836      */
27837     getValue : function(){
27838         return this.value;
27839     },
27840
27841     // private
27842     focus : function(){
27843         if(this.el){
27844             this.update(this.activeDate);
27845         }
27846     },
27847
27848     // privateval
27849     onRender : function(container, position){
27850         
27851         var m = [
27852              '<table cellspacing="0">',
27853                 '<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>',
27854                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27855         var dn = this.dayNames;
27856         for(var i = 0; i < 7; i++){
27857             var d = this.startDay+i;
27858             if(d > 6){
27859                 d = d-7;
27860             }
27861             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27862         }
27863         m[m.length] = "</tr></thead><tbody><tr>";
27864         for(var i = 0; i < 42; i++) {
27865             if(i % 7 == 0 && i != 0){
27866                 m[m.length] = "</tr><tr>";
27867             }
27868             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27869         }
27870         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27871             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27872
27873         var el = document.createElement("div");
27874         el.className = "x-date-picker";
27875         el.innerHTML = m.join("");
27876
27877         container.dom.insertBefore(el, position);
27878
27879         this.el = Roo.get(el);
27880         this.eventEl = Roo.get(el.firstChild);
27881
27882         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27883             handler: this.showPrevMonth,
27884             scope: this,
27885             preventDefault:true,
27886             stopDefault:true
27887         });
27888
27889         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27890             handler: this.showNextMonth,
27891             scope: this,
27892             preventDefault:true,
27893             stopDefault:true
27894         });
27895
27896         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27897
27898         this.monthPicker = this.el.down('div.x-date-mp');
27899         this.monthPicker.enableDisplayMode('block');
27900         
27901         var kn = new Roo.KeyNav(this.eventEl, {
27902             "left" : function(e){
27903                 e.ctrlKey ?
27904                     this.showPrevMonth() :
27905                     this.update(this.activeDate.add("d", -1));
27906             },
27907
27908             "right" : function(e){
27909                 e.ctrlKey ?
27910                     this.showNextMonth() :
27911                     this.update(this.activeDate.add("d", 1));
27912             },
27913
27914             "up" : function(e){
27915                 e.ctrlKey ?
27916                     this.showNextYear() :
27917                     this.update(this.activeDate.add("d", -7));
27918             },
27919
27920             "down" : function(e){
27921                 e.ctrlKey ?
27922                     this.showPrevYear() :
27923                     this.update(this.activeDate.add("d", 7));
27924             },
27925
27926             "pageUp" : function(e){
27927                 this.showNextMonth();
27928             },
27929
27930             "pageDown" : function(e){
27931                 this.showPrevMonth();
27932             },
27933
27934             "enter" : function(e){
27935                 e.stopPropagation();
27936                 return true;
27937             },
27938
27939             scope : this
27940         });
27941
27942         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27943
27944         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27945
27946         this.el.unselectable();
27947         
27948         this.cells = this.el.select("table.x-date-inner tbody td");
27949         this.textNodes = this.el.query("table.x-date-inner tbody span");
27950
27951         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27952             text: "&#160;",
27953             tooltip: this.monthYearText
27954         });
27955
27956         this.mbtn.on('click', this.showMonthPicker, this);
27957         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27958
27959
27960         var today = (new Date()).dateFormat(this.format);
27961         
27962         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
27963         if (this.showClear) {
27964             baseTb.add( new Roo.Toolbar.Fill());
27965         }
27966         baseTb.add({
27967             text: String.format(this.todayText, today),
27968             tooltip: String.format(this.todayTip, today),
27969             handler: this.selectToday,
27970             scope: this
27971         });
27972         
27973         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
27974             
27975         //});
27976         if (this.showClear) {
27977             
27978             baseTb.add( new Roo.Toolbar.Fill());
27979             baseTb.add({
27980                 text: '&#160;',
27981                 cls: 'x-btn-icon x-btn-clear',
27982                 handler: function() {
27983                     //this.value = '';
27984                     this.fireEvent("select", this, '');
27985                 },
27986                 scope: this
27987             });
27988         }
27989         
27990         
27991         if(Roo.isIE){
27992             this.el.repaint();
27993         }
27994         this.update(this.value);
27995     },
27996
27997     createMonthPicker : function(){
27998         if(!this.monthPicker.dom.firstChild){
27999             var buf = ['<table border="0" cellspacing="0">'];
28000             for(var i = 0; i < 6; i++){
28001                 buf.push(
28002                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28003                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28004                     i == 0 ?
28005                     '<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>' :
28006                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28007                 );
28008             }
28009             buf.push(
28010                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28011                     this.okText,
28012                     '</button><button type="button" class="x-date-mp-cancel">',
28013                     this.cancelText,
28014                     '</button></td></tr>',
28015                 '</table>'
28016             );
28017             this.monthPicker.update(buf.join(''));
28018             this.monthPicker.on('click', this.onMonthClick, this);
28019             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28020
28021             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28022             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28023
28024             this.mpMonths.each(function(m, a, i){
28025                 i += 1;
28026                 if((i%2) == 0){
28027                     m.dom.xmonth = 5 + Math.round(i * .5);
28028                 }else{
28029                     m.dom.xmonth = Math.round((i-1) * .5);
28030                 }
28031             });
28032         }
28033     },
28034
28035     showMonthPicker : function(){
28036         this.createMonthPicker();
28037         var size = this.el.getSize();
28038         this.monthPicker.setSize(size);
28039         this.monthPicker.child('table').setSize(size);
28040
28041         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28042         this.updateMPMonth(this.mpSelMonth);
28043         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28044         this.updateMPYear(this.mpSelYear);
28045
28046         this.monthPicker.slideIn('t', {duration:.2});
28047     },
28048
28049     updateMPYear : function(y){
28050         this.mpyear = y;
28051         var ys = this.mpYears.elements;
28052         for(var i = 1; i <= 10; i++){
28053             var td = ys[i-1], y2;
28054             if((i%2) == 0){
28055                 y2 = y + Math.round(i * .5);
28056                 td.firstChild.innerHTML = y2;
28057                 td.xyear = y2;
28058             }else{
28059                 y2 = y - (5-Math.round(i * .5));
28060                 td.firstChild.innerHTML = y2;
28061                 td.xyear = y2;
28062             }
28063             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28064         }
28065     },
28066
28067     updateMPMonth : function(sm){
28068         this.mpMonths.each(function(m, a, i){
28069             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28070         });
28071     },
28072
28073     selectMPMonth: function(m){
28074         
28075     },
28076
28077     onMonthClick : function(e, t){
28078         e.stopEvent();
28079         var el = new Roo.Element(t), pn;
28080         if(el.is('button.x-date-mp-cancel')){
28081             this.hideMonthPicker();
28082         }
28083         else if(el.is('button.x-date-mp-ok')){
28084             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28085             this.hideMonthPicker();
28086         }
28087         else if(pn = el.up('td.x-date-mp-month', 2)){
28088             this.mpMonths.removeClass('x-date-mp-sel');
28089             pn.addClass('x-date-mp-sel');
28090             this.mpSelMonth = pn.dom.xmonth;
28091         }
28092         else if(pn = el.up('td.x-date-mp-year', 2)){
28093             this.mpYears.removeClass('x-date-mp-sel');
28094             pn.addClass('x-date-mp-sel');
28095             this.mpSelYear = pn.dom.xyear;
28096         }
28097         else if(el.is('a.x-date-mp-prev')){
28098             this.updateMPYear(this.mpyear-10);
28099         }
28100         else if(el.is('a.x-date-mp-next')){
28101             this.updateMPYear(this.mpyear+10);
28102         }
28103     },
28104
28105     onMonthDblClick : function(e, t){
28106         e.stopEvent();
28107         var el = new Roo.Element(t), pn;
28108         if(pn = el.up('td.x-date-mp-month', 2)){
28109             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28110             this.hideMonthPicker();
28111         }
28112         else if(pn = el.up('td.x-date-mp-year', 2)){
28113             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28114             this.hideMonthPicker();
28115         }
28116     },
28117
28118     hideMonthPicker : function(disableAnim){
28119         if(this.monthPicker){
28120             if(disableAnim === true){
28121                 this.monthPicker.hide();
28122             }else{
28123                 this.monthPicker.slideOut('t', {duration:.2});
28124             }
28125         }
28126     },
28127
28128     // private
28129     showPrevMonth : function(e){
28130         this.update(this.activeDate.add("mo", -1));
28131     },
28132
28133     // private
28134     showNextMonth : function(e){
28135         this.update(this.activeDate.add("mo", 1));
28136     },
28137
28138     // private
28139     showPrevYear : function(){
28140         this.update(this.activeDate.add("y", -1));
28141     },
28142
28143     // private
28144     showNextYear : function(){
28145         this.update(this.activeDate.add("y", 1));
28146     },
28147
28148     // private
28149     handleMouseWheel : function(e){
28150         var delta = e.getWheelDelta();
28151         if(delta > 0){
28152             this.showPrevMonth();
28153             e.stopEvent();
28154         } else if(delta < 0){
28155             this.showNextMonth();
28156             e.stopEvent();
28157         }
28158     },
28159
28160     // private
28161     handleDateClick : function(e, t){
28162         e.stopEvent();
28163         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28164             this.setValue(new Date(t.dateValue));
28165             this.fireEvent("select", this, this.value);
28166         }
28167     },
28168
28169     // private
28170     selectToday : function(){
28171         this.setValue(new Date().clearTime());
28172         this.fireEvent("select", this, this.value);
28173     },
28174
28175     // private
28176     update : function(date)
28177     {
28178         var vd = this.activeDate;
28179         this.activeDate = date;
28180         if(vd && this.el){
28181             var t = date.getTime();
28182             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28183                 this.cells.removeClass("x-date-selected");
28184                 this.cells.each(function(c){
28185                    if(c.dom.firstChild.dateValue == t){
28186                        c.addClass("x-date-selected");
28187                        setTimeout(function(){
28188                             try{c.dom.firstChild.focus();}catch(e){}
28189                        }, 50);
28190                        return false;
28191                    }
28192                 });
28193                 return;
28194             }
28195         }
28196         
28197         var days = date.getDaysInMonth();
28198         var firstOfMonth = date.getFirstDateOfMonth();
28199         var startingPos = firstOfMonth.getDay()-this.startDay;
28200
28201         if(startingPos <= this.startDay){
28202             startingPos += 7;
28203         }
28204
28205         var pm = date.add("mo", -1);
28206         var prevStart = pm.getDaysInMonth()-startingPos;
28207
28208         var cells = this.cells.elements;
28209         var textEls = this.textNodes;
28210         days += startingPos;
28211
28212         // convert everything to numbers so it's fast
28213         var day = 86400000;
28214         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28215         var today = new Date().clearTime().getTime();
28216         var sel = date.clearTime().getTime();
28217         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28218         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28219         var ddMatch = this.disabledDatesRE;
28220         var ddText = this.disabledDatesText;
28221         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28222         var ddaysText = this.disabledDaysText;
28223         var format = this.format;
28224
28225         var setCellClass = function(cal, cell){
28226             cell.title = "";
28227             var t = d.getTime();
28228             cell.firstChild.dateValue = t;
28229             if(t == today){
28230                 cell.className += " x-date-today";
28231                 cell.title = cal.todayText;
28232             }
28233             if(t == sel){
28234                 cell.className += " x-date-selected";
28235                 setTimeout(function(){
28236                     try{cell.firstChild.focus();}catch(e){}
28237                 }, 50);
28238             }
28239             // disabling
28240             if(t < min) {
28241                 cell.className = " x-date-disabled";
28242                 cell.title = cal.minText;
28243                 return;
28244             }
28245             if(t > max) {
28246                 cell.className = " x-date-disabled";
28247                 cell.title = cal.maxText;
28248                 return;
28249             }
28250             if(ddays){
28251                 if(ddays.indexOf(d.getDay()) != -1){
28252                     cell.title = ddaysText;
28253                     cell.className = " x-date-disabled";
28254                 }
28255             }
28256             if(ddMatch && format){
28257                 var fvalue = d.dateFormat(format);
28258                 if(ddMatch.test(fvalue)){
28259                     cell.title = ddText.replace("%0", fvalue);
28260                     cell.className = " x-date-disabled";
28261                 }
28262             }
28263         };
28264
28265         var i = 0;
28266         for(; i < startingPos; i++) {
28267             textEls[i].innerHTML = (++prevStart);
28268             d.setDate(d.getDate()+1);
28269             cells[i].className = "x-date-prevday";
28270             setCellClass(this, cells[i]);
28271         }
28272         for(; i < days; i++){
28273             intDay = i - startingPos + 1;
28274             textEls[i].innerHTML = (intDay);
28275             d.setDate(d.getDate()+1);
28276             cells[i].className = "x-date-active";
28277             setCellClass(this, cells[i]);
28278         }
28279         var extraDays = 0;
28280         for(; i < 42; i++) {
28281              textEls[i].innerHTML = (++extraDays);
28282              d.setDate(d.getDate()+1);
28283              cells[i].className = "x-date-nextday";
28284              setCellClass(this, cells[i]);
28285         }
28286
28287         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28288         this.fireEvent('monthchange', this, date);
28289         
28290         if(!this.internalRender){
28291             var main = this.el.dom.firstChild;
28292             var w = main.offsetWidth;
28293             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28294             Roo.fly(main).setWidth(w);
28295             this.internalRender = true;
28296             // opera does not respect the auto grow header center column
28297             // then, after it gets a width opera refuses to recalculate
28298             // without a second pass
28299             if(Roo.isOpera && !this.secondPass){
28300                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28301                 this.secondPass = true;
28302                 this.update.defer(10, this, [date]);
28303             }
28304         }
28305         
28306         
28307     }
28308 });        /*
28309  * Based on:
28310  * Ext JS Library 1.1.1
28311  * Copyright(c) 2006-2007, Ext JS, LLC.
28312  *
28313  * Originally Released Under LGPL - original licence link has changed is not relivant.
28314  *
28315  * Fork - LGPL
28316  * <script type="text/javascript">
28317  */
28318 /**
28319  * @class Roo.TabPanel
28320  * @extends Roo.util.Observable
28321  * A lightweight tab container.
28322  * <br><br>
28323  * Usage:
28324  * <pre><code>
28325 // basic tabs 1, built from existing content
28326 var tabs = new Roo.TabPanel("tabs1");
28327 tabs.addTab("script", "View Script");
28328 tabs.addTab("markup", "View Markup");
28329 tabs.activate("script");
28330
28331 // more advanced tabs, built from javascript
28332 var jtabs = new Roo.TabPanel("jtabs");
28333 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28334
28335 // set up the UpdateManager
28336 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28337 var updater = tab2.getUpdateManager();
28338 updater.setDefaultUrl("ajax1.htm");
28339 tab2.on('activate', updater.refresh, updater, true);
28340
28341 // Use setUrl for Ajax loading
28342 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28343 tab3.setUrl("ajax2.htm", null, true);
28344
28345 // Disabled tab
28346 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28347 tab4.disable();
28348
28349 jtabs.activate("jtabs-1");
28350  * </code></pre>
28351  * @constructor
28352  * Create a new TabPanel.
28353  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28354  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28355  */
28356 Roo.TabPanel = function(container, config){
28357     /**
28358     * The container element for this TabPanel.
28359     * @type Roo.Element
28360     */
28361     this.el = Roo.get(container, true);
28362     if(config){
28363         if(typeof config == "boolean"){
28364             this.tabPosition = config ? "bottom" : "top";
28365         }else{
28366             Roo.apply(this, config);
28367         }
28368     }
28369     if(this.tabPosition == "bottom"){
28370         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28371         this.el.addClass("x-tabs-bottom");
28372     }
28373     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28374     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28375     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28376     if(Roo.isIE){
28377         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28378     }
28379     if(this.tabPosition != "bottom"){
28380         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28381          * @type Roo.Element
28382          */
28383         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28384         this.el.addClass("x-tabs-top");
28385     }
28386     this.items = [];
28387
28388     this.bodyEl.setStyle("position", "relative");
28389
28390     this.active = null;
28391     this.activateDelegate = this.activate.createDelegate(this);
28392
28393     this.addEvents({
28394         /**
28395          * @event tabchange
28396          * Fires when the active tab changes
28397          * @param {Roo.TabPanel} this
28398          * @param {Roo.TabPanelItem} activePanel The new active tab
28399          */
28400         "tabchange": true,
28401         /**
28402          * @event beforetabchange
28403          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28404          * @param {Roo.TabPanel} this
28405          * @param {Object} e Set cancel to true on this object to cancel the tab change
28406          * @param {Roo.TabPanelItem} tab The tab being changed to
28407          */
28408         "beforetabchange" : true
28409     });
28410
28411     Roo.EventManager.onWindowResize(this.onResize, this);
28412     this.cpad = this.el.getPadding("lr");
28413     this.hiddenCount = 0;
28414
28415
28416     // toolbar on the tabbar support...
28417     if (this.toolbar) {
28418         var tcfg = this.toolbar;
28419         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28420         this.toolbar = new Roo.Toolbar(tcfg);
28421         if (Roo.isSafari) {
28422             var tbl = tcfg.container.child('table', true);
28423             tbl.setAttribute('width', '100%');
28424         }
28425         
28426     }
28427    
28428
28429
28430     Roo.TabPanel.superclass.constructor.call(this);
28431 };
28432
28433 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28434     /*
28435      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28436      */
28437     tabPosition : "top",
28438     /*
28439      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28440      */
28441     currentTabWidth : 0,
28442     /*
28443      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28444      */
28445     minTabWidth : 40,
28446     /*
28447      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28448      */
28449     maxTabWidth : 250,
28450     /*
28451      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28452      */
28453     preferredTabWidth : 175,
28454     /*
28455      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28456      */
28457     resizeTabs : false,
28458     /*
28459      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28460      */
28461     monitorResize : true,
28462     /*
28463      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28464      */
28465     toolbar : false,
28466
28467     /**
28468      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28469      * @param {String} id The id of the div to use <b>or create</b>
28470      * @param {String} text The text for the tab
28471      * @param {String} content (optional) Content to put in the TabPanelItem body
28472      * @param {Boolean} closable (optional) True to create a close icon on the tab
28473      * @return {Roo.TabPanelItem} The created TabPanelItem
28474      */
28475     addTab : function(id, text, content, closable){
28476         var item = new Roo.TabPanelItem(this, id, text, closable);
28477         this.addTabItem(item);
28478         if(content){
28479             item.setContent(content);
28480         }
28481         return item;
28482     },
28483
28484     /**
28485      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28486      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28487      * @return {Roo.TabPanelItem}
28488      */
28489     getTab : function(id){
28490         return this.items[id];
28491     },
28492
28493     /**
28494      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28495      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28496      */
28497     hideTab : function(id){
28498         var t = this.items[id];
28499         if(!t.isHidden()){
28500            t.setHidden(true);
28501            this.hiddenCount++;
28502            this.autoSizeTabs();
28503         }
28504     },
28505
28506     /**
28507      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28508      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28509      */
28510     unhideTab : function(id){
28511         var t = this.items[id];
28512         if(t.isHidden()){
28513            t.setHidden(false);
28514            this.hiddenCount--;
28515            this.autoSizeTabs();
28516         }
28517     },
28518
28519     /**
28520      * Adds an existing {@link Roo.TabPanelItem}.
28521      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28522      */
28523     addTabItem : function(item){
28524         this.items[item.id] = item;
28525         this.items.push(item);
28526         if(this.resizeTabs){
28527            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28528            this.autoSizeTabs();
28529         }else{
28530             item.autoSize();
28531         }
28532     },
28533
28534     /**
28535      * Removes a {@link Roo.TabPanelItem}.
28536      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28537      */
28538     removeTab : function(id){
28539         var items = this.items;
28540         var tab = items[id];
28541         if(!tab) { return; }
28542         var index = items.indexOf(tab);
28543         if(this.active == tab && items.length > 1){
28544             var newTab = this.getNextAvailable(index);
28545             if(newTab) {
28546                 newTab.activate();
28547             }
28548         }
28549         this.stripEl.dom.removeChild(tab.pnode.dom);
28550         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28551             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28552         }
28553         items.splice(index, 1);
28554         delete this.items[tab.id];
28555         tab.fireEvent("close", tab);
28556         tab.purgeListeners();
28557         this.autoSizeTabs();
28558     },
28559
28560     getNextAvailable : function(start){
28561         var items = this.items;
28562         var index = start;
28563         // look for a next tab that will slide over to
28564         // replace the one being removed
28565         while(index < items.length){
28566             var item = items[++index];
28567             if(item && !item.isHidden()){
28568                 return item;
28569             }
28570         }
28571         // if one isn't found select the previous tab (on the left)
28572         index = start;
28573         while(index >= 0){
28574             var item = items[--index];
28575             if(item && !item.isHidden()){
28576                 return item;
28577             }
28578         }
28579         return null;
28580     },
28581
28582     /**
28583      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28584      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28585      */
28586     disableTab : function(id){
28587         var tab = this.items[id];
28588         if(tab && this.active != tab){
28589             tab.disable();
28590         }
28591     },
28592
28593     /**
28594      * Enables a {@link Roo.TabPanelItem} that is disabled.
28595      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28596      */
28597     enableTab : function(id){
28598         var tab = this.items[id];
28599         tab.enable();
28600     },
28601
28602     /**
28603      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28604      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28605      * @return {Roo.TabPanelItem} The TabPanelItem.
28606      */
28607     activate : function(id){
28608         var tab = this.items[id];
28609         if(!tab){
28610             return null;
28611         }
28612         if(tab == this.active || tab.disabled){
28613             return tab;
28614         }
28615         var e = {};
28616         this.fireEvent("beforetabchange", this, e, tab);
28617         if(e.cancel !== true && !tab.disabled){
28618             if(this.active){
28619                 this.active.hide();
28620             }
28621             this.active = this.items[id];
28622             this.active.show();
28623             this.fireEvent("tabchange", this, this.active);
28624         }
28625         return tab;
28626     },
28627
28628     /**
28629      * Gets the active {@link Roo.TabPanelItem}.
28630      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28631      */
28632     getActiveTab : function(){
28633         return this.active;
28634     },
28635
28636     /**
28637      * Updates the tab body element to fit the height of the container element
28638      * for overflow scrolling
28639      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28640      */
28641     syncHeight : function(targetHeight){
28642         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28643         var bm = this.bodyEl.getMargins();
28644         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28645         this.bodyEl.setHeight(newHeight);
28646         return newHeight;
28647     },
28648
28649     onResize : function(){
28650         if(this.monitorResize){
28651             this.autoSizeTabs();
28652         }
28653     },
28654
28655     /**
28656      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28657      */
28658     beginUpdate : function(){
28659         this.updating = true;
28660     },
28661
28662     /**
28663      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28664      */
28665     endUpdate : function(){
28666         this.updating = false;
28667         this.autoSizeTabs();
28668     },
28669
28670     /**
28671      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28672      */
28673     autoSizeTabs : function(){
28674         var count = this.items.length;
28675         var vcount = count - this.hiddenCount;
28676         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28677             return;
28678         }
28679         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28680         var availWidth = Math.floor(w / vcount);
28681         var b = this.stripBody;
28682         if(b.getWidth() > w){
28683             var tabs = this.items;
28684             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28685             if(availWidth < this.minTabWidth){
28686                 /*if(!this.sleft){    // incomplete scrolling code
28687                     this.createScrollButtons();
28688                 }
28689                 this.showScroll();
28690                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28691             }
28692         }else{
28693             if(this.currentTabWidth < this.preferredTabWidth){
28694                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28695             }
28696         }
28697     },
28698
28699     /**
28700      * Returns the number of tabs in this TabPanel.
28701      * @return {Number}
28702      */
28703      getCount : function(){
28704          return this.items.length;
28705      },
28706
28707     /**
28708      * Resizes all the tabs to the passed width
28709      * @param {Number} The new width
28710      */
28711     setTabWidth : function(width){
28712         this.currentTabWidth = width;
28713         for(var i = 0, len = this.items.length; i < len; i++) {
28714                 if(!this.items[i].isHidden()) {
28715                 this.items[i].setWidth(width);
28716             }
28717         }
28718     },
28719
28720     /**
28721      * Destroys this TabPanel
28722      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28723      */
28724     destroy : function(removeEl){
28725         Roo.EventManager.removeResizeListener(this.onResize, this);
28726         for(var i = 0, len = this.items.length; i < len; i++){
28727             this.items[i].purgeListeners();
28728         }
28729         if(removeEl === true){
28730             this.el.update("");
28731             this.el.remove();
28732         }
28733     }
28734 });
28735
28736 /**
28737  * @class Roo.TabPanelItem
28738  * @extends Roo.util.Observable
28739  * Represents an individual item (tab plus body) in a TabPanel.
28740  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28741  * @param {String} id The id of this TabPanelItem
28742  * @param {String} text The text for the tab of this TabPanelItem
28743  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28744  */
28745 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28746     /**
28747      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28748      * @type Roo.TabPanel
28749      */
28750     this.tabPanel = tabPanel;
28751     /**
28752      * The id for this TabPanelItem
28753      * @type String
28754      */
28755     this.id = id;
28756     /** @private */
28757     this.disabled = false;
28758     /** @private */
28759     this.text = text;
28760     /** @private */
28761     this.loaded = false;
28762     this.closable = closable;
28763
28764     /**
28765      * The body element for this TabPanelItem.
28766      * @type Roo.Element
28767      */
28768     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28769     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28770     this.bodyEl.setStyle("display", "block");
28771     this.bodyEl.setStyle("zoom", "1");
28772     this.hideAction();
28773
28774     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28775     /** @private */
28776     this.el = Roo.get(els.el, true);
28777     this.inner = Roo.get(els.inner, true);
28778     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28779     this.pnode = Roo.get(els.el.parentNode, true);
28780     this.el.on("mousedown", this.onTabMouseDown, this);
28781     this.el.on("click", this.onTabClick, this);
28782     /** @private */
28783     if(closable){
28784         var c = Roo.get(els.close, true);
28785         c.dom.title = this.closeText;
28786         c.addClassOnOver("close-over");
28787         c.on("click", this.closeClick, this);
28788      }
28789
28790     this.addEvents({
28791          /**
28792          * @event activate
28793          * Fires when this tab becomes the active tab.
28794          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28795          * @param {Roo.TabPanelItem} this
28796          */
28797         "activate": true,
28798         /**
28799          * @event beforeclose
28800          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28801          * @param {Roo.TabPanelItem} this
28802          * @param {Object} e Set cancel to true on this object to cancel the close.
28803          */
28804         "beforeclose": true,
28805         /**
28806          * @event close
28807          * Fires when this tab is closed.
28808          * @param {Roo.TabPanelItem} this
28809          */
28810          "close": true,
28811         /**
28812          * @event deactivate
28813          * Fires when this tab is no longer the active tab.
28814          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28815          * @param {Roo.TabPanelItem} this
28816          */
28817          "deactivate" : true
28818     });
28819     this.hidden = false;
28820
28821     Roo.TabPanelItem.superclass.constructor.call(this);
28822 };
28823
28824 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28825     purgeListeners : function(){
28826        Roo.util.Observable.prototype.purgeListeners.call(this);
28827        this.el.removeAllListeners();
28828     },
28829     /**
28830      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28831      */
28832     show : function(){
28833         this.pnode.addClass("on");
28834         this.showAction();
28835         if(Roo.isOpera){
28836             this.tabPanel.stripWrap.repaint();
28837         }
28838         this.fireEvent("activate", this.tabPanel, this);
28839     },
28840
28841     /**
28842      * Returns true if this tab is the active tab.
28843      * @return {Boolean}
28844      */
28845     isActive : function(){
28846         return this.tabPanel.getActiveTab() == this;
28847     },
28848
28849     /**
28850      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28851      */
28852     hide : function(){
28853         this.pnode.removeClass("on");
28854         this.hideAction();
28855         this.fireEvent("deactivate", this.tabPanel, this);
28856     },
28857
28858     hideAction : function(){
28859         this.bodyEl.hide();
28860         this.bodyEl.setStyle("position", "absolute");
28861         this.bodyEl.setLeft("-20000px");
28862         this.bodyEl.setTop("-20000px");
28863     },
28864
28865     showAction : function(){
28866         this.bodyEl.setStyle("position", "relative");
28867         this.bodyEl.setTop("");
28868         this.bodyEl.setLeft("");
28869         this.bodyEl.show();
28870     },
28871
28872     /**
28873      * Set the tooltip for the tab.
28874      * @param {String} tooltip The tab's tooltip
28875      */
28876     setTooltip : function(text){
28877         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28878             this.textEl.dom.qtip = text;
28879             this.textEl.dom.removeAttribute('title');
28880         }else{
28881             this.textEl.dom.title = text;
28882         }
28883     },
28884
28885     onTabClick : function(e){
28886         e.preventDefault();
28887         this.tabPanel.activate(this.id);
28888     },
28889
28890     onTabMouseDown : function(e){
28891         e.preventDefault();
28892         this.tabPanel.activate(this.id);
28893     },
28894
28895     getWidth : function(){
28896         return this.inner.getWidth();
28897     },
28898
28899     setWidth : function(width){
28900         var iwidth = width - this.pnode.getPadding("lr");
28901         this.inner.setWidth(iwidth);
28902         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28903         this.pnode.setWidth(width);
28904     },
28905
28906     /**
28907      * Show or hide the tab
28908      * @param {Boolean} hidden True to hide or false to show.
28909      */
28910     setHidden : function(hidden){
28911         this.hidden = hidden;
28912         this.pnode.setStyle("display", hidden ? "none" : "");
28913     },
28914
28915     /**
28916      * Returns true if this tab is "hidden"
28917      * @return {Boolean}
28918      */
28919     isHidden : function(){
28920         return this.hidden;
28921     },
28922
28923     /**
28924      * Returns the text for this tab
28925      * @return {String}
28926      */
28927     getText : function(){
28928         return this.text;
28929     },
28930
28931     autoSize : function(){
28932         //this.el.beginMeasure();
28933         this.textEl.setWidth(1);
28934         /*
28935          *  #2804 [new] Tabs in Roojs
28936          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28937          */
28938         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28939         //this.el.endMeasure();
28940     },
28941
28942     /**
28943      * Sets the text for the tab (Note: this also sets the tooltip text)
28944      * @param {String} text The tab's text and tooltip
28945      */
28946     setText : function(text){
28947         this.text = text;
28948         this.textEl.update(text);
28949         this.setTooltip(text);
28950         if(!this.tabPanel.resizeTabs){
28951             this.autoSize();
28952         }
28953     },
28954     /**
28955      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28956      */
28957     activate : function(){
28958         this.tabPanel.activate(this.id);
28959     },
28960
28961     /**
28962      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
28963      */
28964     disable : function(){
28965         if(this.tabPanel.active != this){
28966             this.disabled = true;
28967             this.pnode.addClass("disabled");
28968         }
28969     },
28970
28971     /**
28972      * Enables this TabPanelItem if it was previously disabled.
28973      */
28974     enable : function(){
28975         this.disabled = false;
28976         this.pnode.removeClass("disabled");
28977     },
28978
28979     /**
28980      * Sets the content for this TabPanelItem.
28981      * @param {String} content The content
28982      * @param {Boolean} loadScripts true to look for and load scripts
28983      */
28984     setContent : function(content, loadScripts){
28985         this.bodyEl.update(content, loadScripts);
28986     },
28987
28988     /**
28989      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
28990      * @return {Roo.UpdateManager} The UpdateManager
28991      */
28992     getUpdateManager : function(){
28993         return this.bodyEl.getUpdateManager();
28994     },
28995
28996     /**
28997      * Set a URL to be used to load the content for this TabPanelItem.
28998      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
28999      * @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)
29000      * @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)
29001      * @return {Roo.UpdateManager} The UpdateManager
29002      */
29003     setUrl : function(url, params, loadOnce){
29004         if(this.refreshDelegate){
29005             this.un('activate', this.refreshDelegate);
29006         }
29007         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29008         this.on("activate", this.refreshDelegate);
29009         return this.bodyEl.getUpdateManager();
29010     },
29011
29012     /** @private */
29013     _handleRefresh : function(url, params, loadOnce){
29014         if(!loadOnce || !this.loaded){
29015             var updater = this.bodyEl.getUpdateManager();
29016             updater.update(url, params, this._setLoaded.createDelegate(this));
29017         }
29018     },
29019
29020     /**
29021      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29022      *   Will fail silently if the setUrl method has not been called.
29023      *   This does not activate the panel, just updates its content.
29024      */
29025     refresh : function(){
29026         if(this.refreshDelegate){
29027            this.loaded = false;
29028            this.refreshDelegate();
29029         }
29030     },
29031
29032     /** @private */
29033     _setLoaded : function(){
29034         this.loaded = true;
29035     },
29036
29037     /** @private */
29038     closeClick : function(e){
29039         var o = {};
29040         e.stopEvent();
29041         this.fireEvent("beforeclose", this, o);
29042         if(o.cancel !== true){
29043             this.tabPanel.removeTab(this.id);
29044         }
29045     },
29046     /**
29047      * The text displayed in the tooltip for the close icon.
29048      * @type String
29049      */
29050     closeText : "Close this tab"
29051 });
29052
29053 /** @private */
29054 Roo.TabPanel.prototype.createStrip = function(container){
29055     var strip = document.createElement("div");
29056     strip.className = "x-tabs-wrap";
29057     container.appendChild(strip);
29058     return strip;
29059 };
29060 /** @private */
29061 Roo.TabPanel.prototype.createStripList = function(strip){
29062     // div wrapper for retard IE
29063     // returns the "tr" element.
29064     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29065         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29066         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29067     return strip.firstChild.firstChild.firstChild.firstChild;
29068 };
29069 /** @private */
29070 Roo.TabPanel.prototype.createBody = function(container){
29071     var body = document.createElement("div");
29072     Roo.id(body, "tab-body");
29073     Roo.fly(body).addClass("x-tabs-body");
29074     container.appendChild(body);
29075     return body;
29076 };
29077 /** @private */
29078 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29079     var body = Roo.getDom(id);
29080     if(!body){
29081         body = document.createElement("div");
29082         body.id = id;
29083     }
29084     Roo.fly(body).addClass("x-tabs-item-body");
29085     bodyEl.insertBefore(body, bodyEl.firstChild);
29086     return body;
29087 };
29088 /** @private */
29089 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29090     var td = document.createElement("td");
29091     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29092     //stripEl.appendChild(td);
29093     if(closable){
29094         td.className = "x-tabs-closable";
29095         if(!this.closeTpl){
29096             this.closeTpl = new Roo.Template(
29097                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29098                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29099                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29100             );
29101         }
29102         var el = this.closeTpl.overwrite(td, {"text": text});
29103         var close = el.getElementsByTagName("div")[0];
29104         var inner = el.getElementsByTagName("em")[0];
29105         return {"el": el, "close": close, "inner": inner};
29106     } else {
29107         if(!this.tabTpl){
29108             this.tabTpl = new Roo.Template(
29109                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29110                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29111             );
29112         }
29113         var el = this.tabTpl.overwrite(td, {"text": text});
29114         var inner = el.getElementsByTagName("em")[0];
29115         return {"el": el, "inner": inner};
29116     }
29117 };/*
29118  * Based on:
29119  * Ext JS Library 1.1.1
29120  * Copyright(c) 2006-2007, Ext JS, LLC.
29121  *
29122  * Originally Released Under LGPL - original licence link has changed is not relivant.
29123  *
29124  * Fork - LGPL
29125  * <script type="text/javascript">
29126  */
29127
29128 /**
29129  * @class Roo.Button
29130  * @extends Roo.util.Observable
29131  * Simple Button class
29132  * @cfg {String} text The button text
29133  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29134  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29135  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29136  * @cfg {Object} scope The scope of the handler
29137  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29138  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29139  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29140  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29141  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29142  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29143    applies if enableToggle = true)
29144  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29145  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29146   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29147  * @constructor
29148  * Create a new button
29149  * @param {Object} config The config object
29150  */
29151 Roo.Button = function(renderTo, config)
29152 {
29153     if (!config) {
29154         config = renderTo;
29155         renderTo = config.renderTo || false;
29156     }
29157     
29158     Roo.apply(this, config);
29159     this.addEvents({
29160         /**
29161              * @event click
29162              * Fires when this button is clicked
29163              * @param {Button} this
29164              * @param {EventObject} e The click event
29165              */
29166             "click" : true,
29167         /**
29168              * @event toggle
29169              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29170              * @param {Button} this
29171              * @param {Boolean} pressed
29172              */
29173             "toggle" : true,
29174         /**
29175              * @event mouseover
29176              * Fires when the mouse hovers over the button
29177              * @param {Button} this
29178              * @param {Event} e The event object
29179              */
29180         'mouseover' : true,
29181         /**
29182              * @event mouseout
29183              * Fires when the mouse exits the button
29184              * @param {Button} this
29185              * @param {Event} e The event object
29186              */
29187         'mouseout': true,
29188          /**
29189              * @event render
29190              * Fires when the button is rendered
29191              * @param {Button} this
29192              */
29193         'render': true
29194     });
29195     if(this.menu){
29196         this.menu = Roo.menu.MenuMgr.get(this.menu);
29197     }
29198     // register listeners first!!  - so render can be captured..
29199     Roo.util.Observable.call(this);
29200     if(renderTo){
29201         this.render(renderTo);
29202     }
29203     
29204   
29205 };
29206
29207 Roo.extend(Roo.Button, Roo.util.Observable, {
29208     /**
29209      * 
29210      */
29211     
29212     /**
29213      * Read-only. True if this button is hidden
29214      * @type Boolean
29215      */
29216     hidden : false,
29217     /**
29218      * Read-only. True if this button is disabled
29219      * @type Boolean
29220      */
29221     disabled : false,
29222     /**
29223      * Read-only. True if this button is pressed (only if enableToggle = true)
29224      * @type Boolean
29225      */
29226     pressed : false,
29227
29228     /**
29229      * @cfg {Number} tabIndex 
29230      * The DOM tabIndex for this button (defaults to undefined)
29231      */
29232     tabIndex : undefined,
29233
29234     /**
29235      * @cfg {Boolean} enableToggle
29236      * True to enable pressed/not pressed toggling (defaults to false)
29237      */
29238     enableToggle: false,
29239     /**
29240      * @cfg {Mixed} menu
29241      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29242      */
29243     menu : undefined,
29244     /**
29245      * @cfg {String} menuAlign
29246      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29247      */
29248     menuAlign : "tl-bl?",
29249
29250     /**
29251      * @cfg {String} iconCls
29252      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29253      */
29254     iconCls : undefined,
29255     /**
29256      * @cfg {String} type
29257      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29258      */
29259     type : 'button',
29260
29261     // private
29262     menuClassTarget: 'tr',
29263
29264     /**
29265      * @cfg {String} clickEvent
29266      * The type of event to map to the button's event handler (defaults to 'click')
29267      */
29268     clickEvent : 'click',
29269
29270     /**
29271      * @cfg {Boolean} handleMouseEvents
29272      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29273      */
29274     handleMouseEvents : true,
29275
29276     /**
29277      * @cfg {String} tooltipType
29278      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29279      */
29280     tooltipType : 'qtip',
29281
29282     /**
29283      * @cfg {String} cls
29284      * A CSS class to apply to the button's main element.
29285      */
29286     
29287     /**
29288      * @cfg {Roo.Template} template (Optional)
29289      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29290      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29291      * require code modifications if required elements (e.g. a button) aren't present.
29292      */
29293
29294     // private
29295     render : function(renderTo){
29296         var btn;
29297         if(this.hideParent){
29298             this.parentEl = Roo.get(renderTo);
29299         }
29300         if(!this.dhconfig){
29301             if(!this.template){
29302                 if(!Roo.Button.buttonTemplate){
29303                     // hideous table template
29304                     Roo.Button.buttonTemplate = new Roo.Template(
29305                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29306                         '<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>',
29307                         "</tr></tbody></table>");
29308                 }
29309                 this.template = Roo.Button.buttonTemplate;
29310             }
29311             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29312             var btnEl = btn.child("button:first");
29313             btnEl.on('focus', this.onFocus, this);
29314             btnEl.on('blur', this.onBlur, this);
29315             if(this.cls){
29316                 btn.addClass(this.cls);
29317             }
29318             if(this.icon){
29319                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29320             }
29321             if(this.iconCls){
29322                 btnEl.addClass(this.iconCls);
29323                 if(!this.cls){
29324                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29325                 }
29326             }
29327             if(this.tabIndex !== undefined){
29328                 btnEl.dom.tabIndex = this.tabIndex;
29329             }
29330             if(this.tooltip){
29331                 if(typeof this.tooltip == 'object'){
29332                     Roo.QuickTips.tips(Roo.apply({
29333                           target: btnEl.id
29334                     }, this.tooltip));
29335                 } else {
29336                     btnEl.dom[this.tooltipType] = this.tooltip;
29337                 }
29338             }
29339         }else{
29340             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29341         }
29342         this.el = btn;
29343         if(this.id){
29344             this.el.dom.id = this.el.id = this.id;
29345         }
29346         if(this.menu){
29347             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29348             this.menu.on("show", this.onMenuShow, this);
29349             this.menu.on("hide", this.onMenuHide, this);
29350         }
29351         btn.addClass("x-btn");
29352         if(Roo.isIE && !Roo.isIE7){
29353             this.autoWidth.defer(1, this);
29354         }else{
29355             this.autoWidth();
29356         }
29357         if(this.handleMouseEvents){
29358             btn.on("mouseover", this.onMouseOver, this);
29359             btn.on("mouseout", this.onMouseOut, this);
29360             btn.on("mousedown", this.onMouseDown, this);
29361         }
29362         btn.on(this.clickEvent, this.onClick, this);
29363         //btn.on("mouseup", this.onMouseUp, this);
29364         if(this.hidden){
29365             this.hide();
29366         }
29367         if(this.disabled){
29368             this.disable();
29369         }
29370         Roo.ButtonToggleMgr.register(this);
29371         if(this.pressed){
29372             this.el.addClass("x-btn-pressed");
29373         }
29374         if(this.repeat){
29375             var repeater = new Roo.util.ClickRepeater(btn,
29376                 typeof this.repeat == "object" ? this.repeat : {}
29377             );
29378             repeater.on("click", this.onClick,  this);
29379         }
29380         
29381         this.fireEvent('render', this);
29382         
29383     },
29384     /**
29385      * Returns the button's underlying element
29386      * @return {Roo.Element} The element
29387      */
29388     getEl : function(){
29389         return this.el;  
29390     },
29391     
29392     /**
29393      * Destroys this Button and removes any listeners.
29394      */
29395     destroy : function(){
29396         Roo.ButtonToggleMgr.unregister(this);
29397         this.el.removeAllListeners();
29398         this.purgeListeners();
29399         this.el.remove();
29400     },
29401
29402     // private
29403     autoWidth : function(){
29404         if(this.el){
29405             this.el.setWidth("auto");
29406             if(Roo.isIE7 && Roo.isStrict){
29407                 var ib = this.el.child('button');
29408                 if(ib && ib.getWidth() > 20){
29409                     ib.clip();
29410                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29411                 }
29412             }
29413             if(this.minWidth){
29414                 if(this.hidden){
29415                     this.el.beginMeasure();
29416                 }
29417                 if(this.el.getWidth() < this.minWidth){
29418                     this.el.setWidth(this.minWidth);
29419                 }
29420                 if(this.hidden){
29421                     this.el.endMeasure();
29422                 }
29423             }
29424         }
29425     },
29426
29427     /**
29428      * Assigns this button's click handler
29429      * @param {Function} handler The function to call when the button is clicked
29430      * @param {Object} scope (optional) Scope for the function passed in
29431      */
29432     setHandler : function(handler, scope){
29433         this.handler = handler;
29434         this.scope = scope;  
29435     },
29436     
29437     /**
29438      * Sets this button's text
29439      * @param {String} text The button text
29440      */
29441     setText : function(text){
29442         this.text = text;
29443         if(this.el){
29444             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29445         }
29446         this.autoWidth();
29447     },
29448     
29449     /**
29450      * Gets the text for this button
29451      * @return {String} The button text
29452      */
29453     getText : function(){
29454         return this.text;  
29455     },
29456     
29457     /**
29458      * Show this button
29459      */
29460     show: function(){
29461         this.hidden = false;
29462         if(this.el){
29463             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29464         }
29465     },
29466     
29467     /**
29468      * Hide this button
29469      */
29470     hide: function(){
29471         this.hidden = true;
29472         if(this.el){
29473             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29474         }
29475     },
29476     
29477     /**
29478      * Convenience function for boolean show/hide
29479      * @param {Boolean} visible True to show, false to hide
29480      */
29481     setVisible: function(visible){
29482         if(visible) {
29483             this.show();
29484         }else{
29485             this.hide();
29486         }
29487     },
29488     
29489     /**
29490      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29491      * @param {Boolean} state (optional) Force a particular state
29492      */
29493     toggle : function(state){
29494         state = state === undefined ? !this.pressed : state;
29495         if(state != this.pressed){
29496             if(state){
29497                 this.el.addClass("x-btn-pressed");
29498                 this.pressed = true;
29499                 this.fireEvent("toggle", this, true);
29500             }else{
29501                 this.el.removeClass("x-btn-pressed");
29502                 this.pressed = false;
29503                 this.fireEvent("toggle", this, false);
29504             }
29505             if(this.toggleHandler){
29506                 this.toggleHandler.call(this.scope || this, this, state);
29507             }
29508         }
29509     },
29510     
29511     /**
29512      * Focus the button
29513      */
29514     focus : function(){
29515         this.el.child('button:first').focus();
29516     },
29517     
29518     /**
29519      * Disable this button
29520      */
29521     disable : function(){
29522         if(this.el){
29523             this.el.addClass("x-btn-disabled");
29524         }
29525         this.disabled = true;
29526     },
29527     
29528     /**
29529      * Enable this button
29530      */
29531     enable : function(){
29532         if(this.el){
29533             this.el.removeClass("x-btn-disabled");
29534         }
29535         this.disabled = false;
29536     },
29537
29538     /**
29539      * Convenience function for boolean enable/disable
29540      * @param {Boolean} enabled True to enable, false to disable
29541      */
29542     setDisabled : function(v){
29543         this[v !== true ? "enable" : "disable"]();
29544     },
29545
29546     // private
29547     onClick : function(e)
29548     {
29549         if(e){
29550             e.preventDefault();
29551         }
29552         if(e.button != 0){
29553             return;
29554         }
29555         if(!this.disabled){
29556             if(this.enableToggle){
29557                 this.toggle();
29558             }
29559             if(this.menu && !this.menu.isVisible()){
29560                 this.menu.show(this.el, this.menuAlign);
29561             }
29562             this.fireEvent("click", this, e);
29563             if(this.handler){
29564                 this.el.removeClass("x-btn-over");
29565                 this.handler.call(this.scope || this, this, e);
29566             }
29567         }
29568     },
29569     // private
29570     onMouseOver : function(e){
29571         if(!this.disabled){
29572             this.el.addClass("x-btn-over");
29573             this.fireEvent('mouseover', this, e);
29574         }
29575     },
29576     // private
29577     onMouseOut : function(e){
29578         if(!e.within(this.el,  true)){
29579             this.el.removeClass("x-btn-over");
29580             this.fireEvent('mouseout', this, e);
29581         }
29582     },
29583     // private
29584     onFocus : function(e){
29585         if(!this.disabled){
29586             this.el.addClass("x-btn-focus");
29587         }
29588     },
29589     // private
29590     onBlur : function(e){
29591         this.el.removeClass("x-btn-focus");
29592     },
29593     // private
29594     onMouseDown : function(e){
29595         if(!this.disabled && e.button == 0){
29596             this.el.addClass("x-btn-click");
29597             Roo.get(document).on('mouseup', this.onMouseUp, this);
29598         }
29599     },
29600     // private
29601     onMouseUp : function(e){
29602         if(e.button == 0){
29603             this.el.removeClass("x-btn-click");
29604             Roo.get(document).un('mouseup', this.onMouseUp, this);
29605         }
29606     },
29607     // private
29608     onMenuShow : function(e){
29609         this.el.addClass("x-btn-menu-active");
29610     },
29611     // private
29612     onMenuHide : function(e){
29613         this.el.removeClass("x-btn-menu-active");
29614     }   
29615 });
29616
29617 // Private utility class used by Button
29618 Roo.ButtonToggleMgr = function(){
29619    var groups = {};
29620    
29621    function toggleGroup(btn, state){
29622        if(state){
29623            var g = groups[btn.toggleGroup];
29624            for(var i = 0, l = g.length; i < l; i++){
29625                if(g[i] != btn){
29626                    g[i].toggle(false);
29627                }
29628            }
29629        }
29630    }
29631    
29632    return {
29633        register : function(btn){
29634            if(!btn.toggleGroup){
29635                return;
29636            }
29637            var g = groups[btn.toggleGroup];
29638            if(!g){
29639                g = groups[btn.toggleGroup] = [];
29640            }
29641            g.push(btn);
29642            btn.on("toggle", toggleGroup);
29643        },
29644        
29645        unregister : function(btn){
29646            if(!btn.toggleGroup){
29647                return;
29648            }
29649            var g = groups[btn.toggleGroup];
29650            if(g){
29651                g.remove(btn);
29652                btn.un("toggle", toggleGroup);
29653            }
29654        }
29655    };
29656 }();/*
29657  * Based on:
29658  * Ext JS Library 1.1.1
29659  * Copyright(c) 2006-2007, Ext JS, LLC.
29660  *
29661  * Originally Released Under LGPL - original licence link has changed is not relivant.
29662  *
29663  * Fork - LGPL
29664  * <script type="text/javascript">
29665  */
29666  
29667 /**
29668  * @class Roo.SplitButton
29669  * @extends Roo.Button
29670  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29671  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29672  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29673  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29674  * @cfg {String} arrowTooltip The title attribute of the arrow
29675  * @constructor
29676  * Create a new menu button
29677  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29678  * @param {Object} config The config object
29679  */
29680 Roo.SplitButton = function(renderTo, config){
29681     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29682     /**
29683      * @event arrowclick
29684      * Fires when this button's arrow is clicked
29685      * @param {SplitButton} this
29686      * @param {EventObject} e The click event
29687      */
29688     this.addEvents({"arrowclick":true});
29689 };
29690
29691 Roo.extend(Roo.SplitButton, Roo.Button, {
29692     render : function(renderTo){
29693         // this is one sweet looking template!
29694         var tpl = new Roo.Template(
29695             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29696             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29697             '<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>',
29698             "</tbody></table></td><td>",
29699             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29700             '<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>',
29701             "</tbody></table></td></tr></table>"
29702         );
29703         var btn = tpl.append(renderTo, [this.text, this.type], true);
29704         var btnEl = btn.child("button");
29705         if(this.cls){
29706             btn.addClass(this.cls);
29707         }
29708         if(this.icon){
29709             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29710         }
29711         if(this.iconCls){
29712             btnEl.addClass(this.iconCls);
29713             if(!this.cls){
29714                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29715             }
29716         }
29717         this.el = btn;
29718         if(this.handleMouseEvents){
29719             btn.on("mouseover", this.onMouseOver, this);
29720             btn.on("mouseout", this.onMouseOut, this);
29721             btn.on("mousedown", this.onMouseDown, this);
29722             btn.on("mouseup", this.onMouseUp, this);
29723         }
29724         btn.on(this.clickEvent, this.onClick, this);
29725         if(this.tooltip){
29726             if(typeof this.tooltip == 'object'){
29727                 Roo.QuickTips.tips(Roo.apply({
29728                       target: btnEl.id
29729                 }, this.tooltip));
29730             } else {
29731                 btnEl.dom[this.tooltipType] = this.tooltip;
29732             }
29733         }
29734         if(this.arrowTooltip){
29735             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29736         }
29737         if(this.hidden){
29738             this.hide();
29739         }
29740         if(this.disabled){
29741             this.disable();
29742         }
29743         if(this.pressed){
29744             this.el.addClass("x-btn-pressed");
29745         }
29746         if(Roo.isIE && !Roo.isIE7){
29747             this.autoWidth.defer(1, this);
29748         }else{
29749             this.autoWidth();
29750         }
29751         if(this.menu){
29752             this.menu.on("show", this.onMenuShow, this);
29753             this.menu.on("hide", this.onMenuHide, this);
29754         }
29755         this.fireEvent('render', this);
29756     },
29757
29758     // private
29759     autoWidth : function(){
29760         if(this.el){
29761             var tbl = this.el.child("table:first");
29762             var tbl2 = this.el.child("table:last");
29763             this.el.setWidth("auto");
29764             tbl.setWidth("auto");
29765             if(Roo.isIE7 && Roo.isStrict){
29766                 var ib = this.el.child('button:first');
29767                 if(ib && ib.getWidth() > 20){
29768                     ib.clip();
29769                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29770                 }
29771             }
29772             if(this.minWidth){
29773                 if(this.hidden){
29774                     this.el.beginMeasure();
29775                 }
29776                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29777                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29778                 }
29779                 if(this.hidden){
29780                     this.el.endMeasure();
29781                 }
29782             }
29783             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29784         } 
29785     },
29786     /**
29787      * Sets this button's click handler
29788      * @param {Function} handler The function to call when the button is clicked
29789      * @param {Object} scope (optional) Scope for the function passed above
29790      */
29791     setHandler : function(handler, scope){
29792         this.handler = handler;
29793         this.scope = scope;  
29794     },
29795     
29796     /**
29797      * Sets this button's arrow click handler
29798      * @param {Function} handler The function to call when the arrow is clicked
29799      * @param {Object} scope (optional) Scope for the function passed above
29800      */
29801     setArrowHandler : function(handler, scope){
29802         this.arrowHandler = handler;
29803         this.scope = scope;  
29804     },
29805     
29806     /**
29807      * Focus the button
29808      */
29809     focus : function(){
29810         if(this.el){
29811             this.el.child("button:first").focus();
29812         }
29813     },
29814
29815     // private
29816     onClick : function(e){
29817         e.preventDefault();
29818         if(!this.disabled){
29819             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29820                 if(this.menu && !this.menu.isVisible()){
29821                     this.menu.show(this.el, this.menuAlign);
29822                 }
29823                 this.fireEvent("arrowclick", this, e);
29824                 if(this.arrowHandler){
29825                     this.arrowHandler.call(this.scope || this, this, e);
29826                 }
29827             }else{
29828                 this.fireEvent("click", this, e);
29829                 if(this.handler){
29830                     this.handler.call(this.scope || this, this, e);
29831                 }
29832             }
29833         }
29834     },
29835     // private
29836     onMouseDown : function(e){
29837         if(!this.disabled){
29838             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29839         }
29840     },
29841     // private
29842     onMouseUp : function(e){
29843         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29844     }   
29845 });
29846
29847
29848 // backwards compat
29849 Roo.MenuButton = Roo.SplitButton;/*
29850  * Based on:
29851  * Ext JS Library 1.1.1
29852  * Copyright(c) 2006-2007, Ext JS, LLC.
29853  *
29854  * Originally Released Under LGPL - original licence link has changed is not relivant.
29855  *
29856  * Fork - LGPL
29857  * <script type="text/javascript">
29858  */
29859
29860 /**
29861  * @class Roo.Toolbar
29862  * Basic Toolbar class.
29863  * @constructor
29864  * Creates a new Toolbar
29865  * @param {Object} container The config object
29866  */ 
29867 Roo.Toolbar = function(container, buttons, config)
29868 {
29869     /// old consturctor format still supported..
29870     if(container instanceof Array){ // omit the container for later rendering
29871         buttons = container;
29872         config = buttons;
29873         container = null;
29874     }
29875     if (typeof(container) == 'object' && container.xtype) {
29876         config = container;
29877         container = config.container;
29878         buttons = config.buttons || []; // not really - use items!!
29879     }
29880     var xitems = [];
29881     if (config && config.items) {
29882         xitems = config.items;
29883         delete config.items;
29884     }
29885     Roo.apply(this, config);
29886     this.buttons = buttons;
29887     
29888     if(container){
29889         this.render(container);
29890     }
29891     this.xitems = xitems;
29892     Roo.each(xitems, function(b) {
29893         this.add(b);
29894     }, this);
29895     
29896 };
29897
29898 Roo.Toolbar.prototype = {
29899     /**
29900      * @cfg {Array} items
29901      * array of button configs or elements to add (will be converted to a MixedCollection)
29902      */
29903     
29904     /**
29905      * @cfg {String/HTMLElement/Element} container
29906      * The id or element that will contain the toolbar
29907      */
29908     // private
29909     render : function(ct){
29910         this.el = Roo.get(ct);
29911         if(this.cls){
29912             this.el.addClass(this.cls);
29913         }
29914         // using a table allows for vertical alignment
29915         // 100% width is needed by Safari...
29916         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29917         this.tr = this.el.child("tr", true);
29918         var autoId = 0;
29919         this.items = new Roo.util.MixedCollection(false, function(o){
29920             return o.id || ("item" + (++autoId));
29921         });
29922         if(this.buttons){
29923             this.add.apply(this, this.buttons);
29924             delete this.buttons;
29925         }
29926     },
29927
29928     /**
29929      * Adds element(s) to the toolbar -- this function takes a variable number of 
29930      * arguments of mixed type and adds them to the toolbar.
29931      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29932      * <ul>
29933      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29934      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29935      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29936      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29937      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29938      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29939      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29940      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29941      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29942      * </ul>
29943      * @param {Mixed} arg2
29944      * @param {Mixed} etc.
29945      */
29946     add : function(){
29947         var a = arguments, l = a.length;
29948         for(var i = 0; i < l; i++){
29949             this._add(a[i]);
29950         }
29951     },
29952     // private..
29953     _add : function(el) {
29954         
29955         if (el.xtype) {
29956             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29957         }
29958         
29959         if (el.applyTo){ // some kind of form field
29960             return this.addField(el);
29961         } 
29962         if (el.render){ // some kind of Toolbar.Item
29963             return this.addItem(el);
29964         }
29965         if (typeof el == "string"){ // string
29966             if(el == "separator" || el == "-"){
29967                 return this.addSeparator();
29968             }
29969             if (el == " "){
29970                 return this.addSpacer();
29971             }
29972             if(el == "->"){
29973                 return this.addFill();
29974             }
29975             return this.addText(el);
29976             
29977         }
29978         if(el.tagName){ // element
29979             return this.addElement(el);
29980         }
29981         if(typeof el == "object"){ // must be button config?
29982             return this.addButton(el);
29983         }
29984         // and now what?!?!
29985         return false;
29986         
29987     },
29988     
29989     /**
29990      * Add an Xtype element
29991      * @param {Object} xtype Xtype Object
29992      * @return {Object} created Object
29993      */
29994     addxtype : function(e){
29995         return this.add(e);  
29996     },
29997     
29998     /**
29999      * Returns the Element for this toolbar.
30000      * @return {Roo.Element}
30001      */
30002     getEl : function(){
30003         return this.el;  
30004     },
30005     
30006     /**
30007      * Adds a separator
30008      * @return {Roo.Toolbar.Item} The separator item
30009      */
30010     addSeparator : function(){
30011         return this.addItem(new Roo.Toolbar.Separator());
30012     },
30013
30014     /**
30015      * Adds a spacer element
30016      * @return {Roo.Toolbar.Spacer} The spacer item
30017      */
30018     addSpacer : function(){
30019         return this.addItem(new Roo.Toolbar.Spacer());
30020     },
30021
30022     /**
30023      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30024      * @return {Roo.Toolbar.Fill} The fill item
30025      */
30026     addFill : function(){
30027         return this.addItem(new Roo.Toolbar.Fill());
30028     },
30029
30030     /**
30031      * Adds any standard HTML element to the toolbar
30032      * @param {String/HTMLElement/Element} el The element or id of the element to add
30033      * @return {Roo.Toolbar.Item} The element's item
30034      */
30035     addElement : function(el){
30036         return this.addItem(new Roo.Toolbar.Item(el));
30037     },
30038     /**
30039      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30040      * @type Roo.util.MixedCollection  
30041      */
30042     items : false,
30043      
30044     /**
30045      * Adds any Toolbar.Item or subclass
30046      * @param {Roo.Toolbar.Item} item
30047      * @return {Roo.Toolbar.Item} The item
30048      */
30049     addItem : function(item){
30050         var td = this.nextBlock();
30051         item.render(td);
30052         this.items.add(item);
30053         return item;
30054     },
30055     
30056     /**
30057      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30058      * @param {Object/Array} config A button config or array of configs
30059      * @return {Roo.Toolbar.Button/Array}
30060      */
30061     addButton : function(config){
30062         if(config instanceof Array){
30063             var buttons = [];
30064             for(var i = 0, len = config.length; i < len; i++) {
30065                 buttons.push(this.addButton(config[i]));
30066             }
30067             return buttons;
30068         }
30069         var b = config;
30070         if(!(config instanceof Roo.Toolbar.Button)){
30071             b = config.split ?
30072                 new Roo.Toolbar.SplitButton(config) :
30073                 new Roo.Toolbar.Button(config);
30074         }
30075         var td = this.nextBlock();
30076         b.render(td);
30077         this.items.add(b);
30078         return b;
30079     },
30080     
30081     /**
30082      * Adds text to the toolbar
30083      * @param {String} text The text to add
30084      * @return {Roo.Toolbar.Item} The element's item
30085      */
30086     addText : function(text){
30087         return this.addItem(new Roo.Toolbar.TextItem(text));
30088     },
30089     
30090     /**
30091      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30092      * @param {Number} index The index where the item is to be inserted
30093      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30094      * @return {Roo.Toolbar.Button/Item}
30095      */
30096     insertButton : function(index, item){
30097         if(item instanceof Array){
30098             var buttons = [];
30099             for(var i = 0, len = item.length; i < len; i++) {
30100                buttons.push(this.insertButton(index + i, item[i]));
30101             }
30102             return buttons;
30103         }
30104         if (!(item instanceof Roo.Toolbar.Button)){
30105            item = new Roo.Toolbar.Button(item);
30106         }
30107         var td = document.createElement("td");
30108         this.tr.insertBefore(td, this.tr.childNodes[index]);
30109         item.render(td);
30110         this.items.insert(index, item);
30111         return item;
30112     },
30113     
30114     /**
30115      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30116      * @param {Object} config
30117      * @return {Roo.Toolbar.Item} The element's item
30118      */
30119     addDom : function(config, returnEl){
30120         var td = this.nextBlock();
30121         Roo.DomHelper.overwrite(td, config);
30122         var ti = new Roo.Toolbar.Item(td.firstChild);
30123         ti.render(td);
30124         this.items.add(ti);
30125         return ti;
30126     },
30127
30128     /**
30129      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30130      * @type Roo.util.MixedCollection  
30131      */
30132     fields : false,
30133     
30134     /**
30135      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30136      * Note: the field should not have been rendered yet. For a field that has already been
30137      * rendered, use {@link #addElement}.
30138      * @param {Roo.form.Field} field
30139      * @return {Roo.ToolbarItem}
30140      */
30141      
30142       
30143     addField : function(field) {
30144         if (!this.fields) {
30145             var autoId = 0;
30146             this.fields = new Roo.util.MixedCollection(false, function(o){
30147                 return o.id || ("item" + (++autoId));
30148             });
30149
30150         }
30151         
30152         var td = this.nextBlock();
30153         field.render(td);
30154         var ti = new Roo.Toolbar.Item(td.firstChild);
30155         ti.render(td);
30156         this.items.add(ti);
30157         this.fields.add(field);
30158         return ti;
30159     },
30160     /**
30161      * Hide the toolbar
30162      * @method hide
30163      */
30164      
30165       
30166     hide : function()
30167     {
30168         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30169         this.el.child('div').hide();
30170     },
30171     /**
30172      * Show the toolbar
30173      * @method show
30174      */
30175     show : function()
30176     {
30177         this.el.child('div').show();
30178     },
30179       
30180     // private
30181     nextBlock : function(){
30182         var td = document.createElement("td");
30183         this.tr.appendChild(td);
30184         return td;
30185     },
30186
30187     // private
30188     destroy : function(){
30189         if(this.items){ // rendered?
30190             Roo.destroy.apply(Roo, this.items.items);
30191         }
30192         if(this.fields){ // rendered?
30193             Roo.destroy.apply(Roo, this.fields.items);
30194         }
30195         Roo.Element.uncache(this.el, this.tr);
30196     }
30197 };
30198
30199 /**
30200  * @class Roo.Toolbar.Item
30201  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30202  * @constructor
30203  * Creates a new Item
30204  * @param {HTMLElement} el 
30205  */
30206 Roo.Toolbar.Item = function(el){
30207     var cfg = {};
30208     if (typeof (el.xtype) != 'undefined') {
30209         cfg = el;
30210         el = cfg.el;
30211     }
30212     
30213     this.el = Roo.getDom(el);
30214     this.id = Roo.id(this.el);
30215     this.hidden = false;
30216     
30217     this.addEvents({
30218          /**
30219              * @event render
30220              * Fires when the button is rendered
30221              * @param {Button} this
30222              */
30223         'render': true
30224     });
30225     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30226 };
30227 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30228 //Roo.Toolbar.Item.prototype = {
30229     
30230     /**
30231      * Get this item's HTML Element
30232      * @return {HTMLElement}
30233      */
30234     getEl : function(){
30235        return this.el;  
30236     },
30237
30238     // private
30239     render : function(td){
30240         
30241          this.td = td;
30242         td.appendChild(this.el);
30243         
30244         this.fireEvent('render', this);
30245     },
30246     
30247     /**
30248      * Removes and destroys this item.
30249      */
30250     destroy : function(){
30251         this.td.parentNode.removeChild(this.td);
30252     },
30253     
30254     /**
30255      * Shows this item.
30256      */
30257     show: function(){
30258         this.hidden = false;
30259         this.td.style.display = "";
30260     },
30261     
30262     /**
30263      * Hides this item.
30264      */
30265     hide: function(){
30266         this.hidden = true;
30267         this.td.style.display = "none";
30268     },
30269     
30270     /**
30271      * Convenience function for boolean show/hide.
30272      * @param {Boolean} visible true to show/false to hide
30273      */
30274     setVisible: function(visible){
30275         if(visible) {
30276             this.show();
30277         }else{
30278             this.hide();
30279         }
30280     },
30281     
30282     /**
30283      * Try to focus this item.
30284      */
30285     focus : function(){
30286         Roo.fly(this.el).focus();
30287     },
30288     
30289     /**
30290      * Disables this item.
30291      */
30292     disable : function(){
30293         Roo.fly(this.td).addClass("x-item-disabled");
30294         this.disabled = true;
30295         this.el.disabled = true;
30296     },
30297     
30298     /**
30299      * Enables this item.
30300      */
30301     enable : function(){
30302         Roo.fly(this.td).removeClass("x-item-disabled");
30303         this.disabled = false;
30304         this.el.disabled = false;
30305     }
30306 });
30307
30308
30309 /**
30310  * @class Roo.Toolbar.Separator
30311  * @extends Roo.Toolbar.Item
30312  * A simple toolbar separator class
30313  * @constructor
30314  * Creates a new Separator
30315  */
30316 Roo.Toolbar.Separator = function(cfg){
30317     
30318     var s = document.createElement("span");
30319     s.className = "ytb-sep";
30320     if (cfg) {
30321         cfg.el = s;
30322     }
30323     
30324     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30325 };
30326 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30327     enable:Roo.emptyFn,
30328     disable:Roo.emptyFn,
30329     focus:Roo.emptyFn
30330 });
30331
30332 /**
30333  * @class Roo.Toolbar.Spacer
30334  * @extends Roo.Toolbar.Item
30335  * A simple element that adds extra horizontal space to a toolbar.
30336  * @constructor
30337  * Creates a new Spacer
30338  */
30339 Roo.Toolbar.Spacer = function(cfg){
30340     var s = document.createElement("div");
30341     s.className = "ytb-spacer";
30342     if (cfg) {
30343         cfg.el = s;
30344     }
30345     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30346 };
30347 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30348     enable:Roo.emptyFn,
30349     disable:Roo.emptyFn,
30350     focus:Roo.emptyFn
30351 });
30352
30353 /**
30354  * @class Roo.Toolbar.Fill
30355  * @extends Roo.Toolbar.Spacer
30356  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30357  * @constructor
30358  * Creates a new Spacer
30359  */
30360 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30361     // private
30362     render : function(td){
30363         td.style.width = '100%';
30364         Roo.Toolbar.Fill.superclass.render.call(this, td);
30365     }
30366 });
30367
30368 /**
30369  * @class Roo.Toolbar.TextItem
30370  * @extends Roo.Toolbar.Item
30371  * A simple class that renders text directly into a toolbar.
30372  * @constructor
30373  * Creates a new TextItem
30374  * @param {String} text
30375  */
30376 Roo.Toolbar.TextItem = function(cfg){
30377     var  text = cfg || "";
30378     if (typeof(cfg) == 'object') {
30379         text = cfg.text || "";
30380     }  else {
30381         cfg = null;
30382     }
30383     var s = document.createElement("span");
30384     s.className = "ytb-text";
30385     s.innerHTML = text;
30386     if (cfg) {
30387         cfg.el  = s;
30388     }
30389     
30390     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30391 };
30392 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30393     
30394      
30395     enable:Roo.emptyFn,
30396     disable:Roo.emptyFn,
30397     focus:Roo.emptyFn
30398 });
30399
30400 /**
30401  * @class Roo.Toolbar.Button
30402  * @extends Roo.Button
30403  * A button that renders into a toolbar.
30404  * @constructor
30405  * Creates a new Button
30406  * @param {Object} config A standard {@link Roo.Button} config object
30407  */
30408 Roo.Toolbar.Button = function(config){
30409     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30410 };
30411 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30412     render : function(td){
30413         this.td = td;
30414         Roo.Toolbar.Button.superclass.render.call(this, td);
30415     },
30416     
30417     /**
30418      * Removes and destroys this button
30419      */
30420     destroy : function(){
30421         Roo.Toolbar.Button.superclass.destroy.call(this);
30422         this.td.parentNode.removeChild(this.td);
30423     },
30424     
30425     /**
30426      * Shows this button
30427      */
30428     show: function(){
30429         this.hidden = false;
30430         this.td.style.display = "";
30431     },
30432     
30433     /**
30434      * Hides this button
30435      */
30436     hide: function(){
30437         this.hidden = true;
30438         this.td.style.display = "none";
30439     },
30440
30441     /**
30442      * Disables this item
30443      */
30444     disable : function(){
30445         Roo.fly(this.td).addClass("x-item-disabled");
30446         this.disabled = true;
30447     },
30448
30449     /**
30450      * Enables this item
30451      */
30452     enable : function(){
30453         Roo.fly(this.td).removeClass("x-item-disabled");
30454         this.disabled = false;
30455     }
30456 });
30457 // backwards compat
30458 Roo.ToolbarButton = Roo.Toolbar.Button;
30459
30460 /**
30461  * @class Roo.Toolbar.SplitButton
30462  * @extends Roo.SplitButton
30463  * A menu button that renders into a toolbar.
30464  * @constructor
30465  * Creates a new SplitButton
30466  * @param {Object} config A standard {@link Roo.SplitButton} config object
30467  */
30468 Roo.Toolbar.SplitButton = function(config){
30469     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30470 };
30471 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30472     render : function(td){
30473         this.td = td;
30474         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30475     },
30476     
30477     /**
30478      * Removes and destroys this button
30479      */
30480     destroy : function(){
30481         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30482         this.td.parentNode.removeChild(this.td);
30483     },
30484     
30485     /**
30486      * Shows this button
30487      */
30488     show: function(){
30489         this.hidden = false;
30490         this.td.style.display = "";
30491     },
30492     
30493     /**
30494      * Hides this button
30495      */
30496     hide: function(){
30497         this.hidden = true;
30498         this.td.style.display = "none";
30499     }
30500 });
30501
30502 // backwards compat
30503 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30504  * Based on:
30505  * Ext JS Library 1.1.1
30506  * Copyright(c) 2006-2007, Ext JS, LLC.
30507  *
30508  * Originally Released Under LGPL - original licence link has changed is not relivant.
30509  *
30510  * Fork - LGPL
30511  * <script type="text/javascript">
30512  */
30513  
30514 /**
30515  * @class Roo.PagingToolbar
30516  * @extends Roo.Toolbar
30517  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30518  * @constructor
30519  * Create a new PagingToolbar
30520  * @param {Object} config The config object
30521  */
30522 Roo.PagingToolbar = function(el, ds, config)
30523 {
30524     // old args format still supported... - xtype is prefered..
30525     if (typeof(el) == 'object' && el.xtype) {
30526         // created from xtype...
30527         config = el;
30528         ds = el.dataSource;
30529         el = config.container;
30530     }
30531     var items = [];
30532     if (config.items) {
30533         items = config.items;
30534         config.items = [];
30535     }
30536     
30537     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30538     this.ds = ds;
30539     this.cursor = 0;
30540     this.renderButtons(this.el);
30541     this.bind(ds);
30542     
30543     // supprot items array.
30544    
30545     Roo.each(items, function(e) {
30546         this.add(Roo.factory(e));
30547     },this);
30548     
30549 };
30550
30551 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30552     /**
30553      * @cfg {Roo.data.Store} dataSource
30554      * The underlying data store providing the paged data
30555      */
30556     /**
30557      * @cfg {String/HTMLElement/Element} container
30558      * container The id or element that will contain the toolbar
30559      */
30560     /**
30561      * @cfg {Boolean} displayInfo
30562      * True to display the displayMsg (defaults to false)
30563      */
30564     /**
30565      * @cfg {Number} pageSize
30566      * The number of records to display per page (defaults to 20)
30567      */
30568     pageSize: 20,
30569     /**
30570      * @cfg {String} displayMsg
30571      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30572      */
30573     displayMsg : 'Displaying {0} - {1} of {2}',
30574     /**
30575      * @cfg {String} emptyMsg
30576      * The message to display when no records are found (defaults to "No data to display")
30577      */
30578     emptyMsg : 'No data to display',
30579     /**
30580      * Customizable piece of the default paging text (defaults to "Page")
30581      * @type String
30582      */
30583     beforePageText : "Page",
30584     /**
30585      * Customizable piece of the default paging text (defaults to "of %0")
30586      * @type String
30587      */
30588     afterPageText : "of {0}",
30589     /**
30590      * Customizable piece of the default paging text (defaults to "First Page")
30591      * @type String
30592      */
30593     firstText : "First Page",
30594     /**
30595      * Customizable piece of the default paging text (defaults to "Previous Page")
30596      * @type String
30597      */
30598     prevText : "Previous Page",
30599     /**
30600      * Customizable piece of the default paging text (defaults to "Next Page")
30601      * @type String
30602      */
30603     nextText : "Next Page",
30604     /**
30605      * Customizable piece of the default paging text (defaults to "Last Page")
30606      * @type String
30607      */
30608     lastText : "Last Page",
30609     /**
30610      * Customizable piece of the default paging text (defaults to "Refresh")
30611      * @type String
30612      */
30613     refreshText : "Refresh",
30614
30615     // private
30616     renderButtons : function(el){
30617         Roo.PagingToolbar.superclass.render.call(this, el);
30618         this.first = this.addButton({
30619             tooltip: this.firstText,
30620             cls: "x-btn-icon x-grid-page-first",
30621             disabled: true,
30622             handler: this.onClick.createDelegate(this, ["first"])
30623         });
30624         this.prev = this.addButton({
30625             tooltip: this.prevText,
30626             cls: "x-btn-icon x-grid-page-prev",
30627             disabled: true,
30628             handler: this.onClick.createDelegate(this, ["prev"])
30629         });
30630         //this.addSeparator();
30631         this.add(this.beforePageText);
30632         this.field = Roo.get(this.addDom({
30633            tag: "input",
30634            type: "text",
30635            size: "3",
30636            value: "1",
30637            cls: "x-grid-page-number"
30638         }).el);
30639         this.field.on("keydown", this.onPagingKeydown, this);
30640         this.field.on("focus", function(){this.dom.select();});
30641         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30642         this.field.setHeight(18);
30643         //this.addSeparator();
30644         this.next = this.addButton({
30645             tooltip: this.nextText,
30646             cls: "x-btn-icon x-grid-page-next",
30647             disabled: true,
30648             handler: this.onClick.createDelegate(this, ["next"])
30649         });
30650         this.last = this.addButton({
30651             tooltip: this.lastText,
30652             cls: "x-btn-icon x-grid-page-last",
30653             disabled: true,
30654             handler: this.onClick.createDelegate(this, ["last"])
30655         });
30656         //this.addSeparator();
30657         this.loading = this.addButton({
30658             tooltip: this.refreshText,
30659             cls: "x-btn-icon x-grid-loading",
30660             handler: this.onClick.createDelegate(this, ["refresh"])
30661         });
30662
30663         if(this.displayInfo){
30664             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30665         }
30666     },
30667
30668     // private
30669     updateInfo : function(){
30670         if(this.displayEl){
30671             var count = this.ds.getCount();
30672             var msg = count == 0 ?
30673                 this.emptyMsg :
30674                 String.format(
30675                     this.displayMsg,
30676                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30677                 );
30678             this.displayEl.update(msg);
30679         }
30680     },
30681
30682     // private
30683     onLoad : function(ds, r, o){
30684        this.cursor = o.params ? o.params.start : 0;
30685        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30686
30687        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30688        this.field.dom.value = ap;
30689        this.first.setDisabled(ap == 1);
30690        this.prev.setDisabled(ap == 1);
30691        this.next.setDisabled(ap == ps);
30692        this.last.setDisabled(ap == ps);
30693        this.loading.enable();
30694        this.updateInfo();
30695     },
30696
30697     // private
30698     getPageData : function(){
30699         var total = this.ds.getTotalCount();
30700         return {
30701             total : total,
30702             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30703             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30704         };
30705     },
30706
30707     // private
30708     onLoadError : function(){
30709         this.loading.enable();
30710     },
30711
30712     // private
30713     onPagingKeydown : function(e){
30714         var k = e.getKey();
30715         var d = this.getPageData();
30716         if(k == e.RETURN){
30717             var v = this.field.dom.value, pageNum;
30718             if(!v || isNaN(pageNum = parseInt(v, 10))){
30719                 this.field.dom.value = d.activePage;
30720                 return;
30721             }
30722             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30723             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30724             e.stopEvent();
30725         }
30726         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))
30727         {
30728           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30729           this.field.dom.value = pageNum;
30730           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30731           e.stopEvent();
30732         }
30733         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30734         {
30735           var v = this.field.dom.value, pageNum; 
30736           var increment = (e.shiftKey) ? 10 : 1;
30737           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30738             increment *= -1;
30739           }
30740           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30741             this.field.dom.value = d.activePage;
30742             return;
30743           }
30744           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30745           {
30746             this.field.dom.value = parseInt(v, 10) + increment;
30747             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30748             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30749           }
30750           e.stopEvent();
30751         }
30752     },
30753
30754     // private
30755     beforeLoad : function(){
30756         if(this.loading){
30757             this.loading.disable();
30758         }
30759     },
30760
30761     // private
30762     onClick : function(which){
30763         var ds = this.ds;
30764         switch(which){
30765             case "first":
30766                 ds.load({params:{start: 0, limit: this.pageSize}});
30767             break;
30768             case "prev":
30769                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30770             break;
30771             case "next":
30772                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30773             break;
30774             case "last":
30775                 var total = ds.getTotalCount();
30776                 var extra = total % this.pageSize;
30777                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30778                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30779             break;
30780             case "refresh":
30781                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30782             break;
30783         }
30784     },
30785
30786     /**
30787      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30788      * @param {Roo.data.Store} store The data store to unbind
30789      */
30790     unbind : function(ds){
30791         ds.un("beforeload", this.beforeLoad, this);
30792         ds.un("load", this.onLoad, this);
30793         ds.un("loadexception", this.onLoadError, this);
30794         ds.un("remove", this.updateInfo, this);
30795         ds.un("add", this.updateInfo, this);
30796         this.ds = undefined;
30797     },
30798
30799     /**
30800      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30801      * @param {Roo.data.Store} store The data store to bind
30802      */
30803     bind : function(ds){
30804         ds.on("beforeload", this.beforeLoad, this);
30805         ds.on("load", this.onLoad, this);
30806         ds.on("loadexception", this.onLoadError, this);
30807         ds.on("remove", this.updateInfo, this);
30808         ds.on("add", this.updateInfo, this);
30809         this.ds = ds;
30810     }
30811 });/*
30812  * Based on:
30813  * Ext JS Library 1.1.1
30814  * Copyright(c) 2006-2007, Ext JS, LLC.
30815  *
30816  * Originally Released Under LGPL - original licence link has changed is not relivant.
30817  *
30818  * Fork - LGPL
30819  * <script type="text/javascript">
30820  */
30821
30822 /**
30823  * @class Roo.Resizable
30824  * @extends Roo.util.Observable
30825  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30826  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30827  * 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
30828  * the element will be wrapped for you automatically.</p>
30829  * <p>Here is the list of valid resize handles:</p>
30830  * <pre>
30831 Value   Description
30832 ------  -------------------
30833  'n'     north
30834  's'     south
30835  'e'     east
30836  'w'     west
30837  'nw'    northwest
30838  'sw'    southwest
30839  'se'    southeast
30840  'ne'    northeast
30841  'hd'    horizontal drag
30842  'all'   all
30843 </pre>
30844  * <p>Here's an example showing the creation of a typical Resizable:</p>
30845  * <pre><code>
30846 var resizer = new Roo.Resizable("element-id", {
30847     handles: 'all',
30848     minWidth: 200,
30849     minHeight: 100,
30850     maxWidth: 500,
30851     maxHeight: 400,
30852     pinned: true
30853 });
30854 resizer.on("resize", myHandler);
30855 </code></pre>
30856  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30857  * resizer.east.setDisplayed(false);</p>
30858  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30859  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30860  * resize operation's new size (defaults to [0, 0])
30861  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30862  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30863  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30864  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30865  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30866  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30867  * @cfg {Number} width The width of the element in pixels (defaults to null)
30868  * @cfg {Number} height The height of the element in pixels (defaults to null)
30869  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30870  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30871  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30872  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30873  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30874  * in favor of the handles config option (defaults to false)
30875  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30876  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30877  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30878  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30879  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30880  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30881  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30882  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30883  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30884  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30885  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30886  * @constructor
30887  * Create a new resizable component
30888  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30889  * @param {Object} config configuration options
30890   */
30891 Roo.Resizable = function(el, config)
30892 {
30893     this.el = Roo.get(el);
30894
30895     if(config && config.wrap){
30896         config.resizeChild = this.el;
30897         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30898         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30899         this.el.setStyle("overflow", "hidden");
30900         this.el.setPositioning(config.resizeChild.getPositioning());
30901         config.resizeChild.clearPositioning();
30902         if(!config.width || !config.height){
30903             var csize = config.resizeChild.getSize();
30904             this.el.setSize(csize.width, csize.height);
30905         }
30906         if(config.pinned && !config.adjustments){
30907             config.adjustments = "auto";
30908         }
30909     }
30910
30911     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30912     this.proxy.unselectable();
30913     this.proxy.enableDisplayMode('block');
30914
30915     Roo.apply(this, config);
30916
30917     if(this.pinned){
30918         this.disableTrackOver = true;
30919         this.el.addClass("x-resizable-pinned");
30920     }
30921     // if the element isn't positioned, make it relative
30922     var position = this.el.getStyle("position");
30923     if(position != "absolute" && position != "fixed"){
30924         this.el.setStyle("position", "relative");
30925     }
30926     if(!this.handles){ // no handles passed, must be legacy style
30927         this.handles = 's,e,se';
30928         if(this.multiDirectional){
30929             this.handles += ',n,w';
30930         }
30931     }
30932     if(this.handles == "all"){
30933         this.handles = "n s e w ne nw se sw";
30934     }
30935     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30936     var ps = Roo.Resizable.positions;
30937     for(var i = 0, len = hs.length; i < len; i++){
30938         if(hs[i] && ps[hs[i]]){
30939             var pos = ps[hs[i]];
30940             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30941         }
30942     }
30943     // legacy
30944     this.corner = this.southeast;
30945     
30946     // updateBox = the box can move..
30947     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30948         this.updateBox = true;
30949     }
30950
30951     this.activeHandle = null;
30952
30953     if(this.resizeChild){
30954         if(typeof this.resizeChild == "boolean"){
30955             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30956         }else{
30957             this.resizeChild = Roo.get(this.resizeChild, true);
30958         }
30959     }
30960     
30961     if(this.adjustments == "auto"){
30962         var rc = this.resizeChild;
30963         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
30964         if(rc && (hw || hn)){
30965             rc.position("relative");
30966             rc.setLeft(hw ? hw.el.getWidth() : 0);
30967             rc.setTop(hn ? hn.el.getHeight() : 0);
30968         }
30969         this.adjustments = [
30970             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
30971             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
30972         ];
30973     }
30974
30975     if(this.draggable){
30976         this.dd = this.dynamic ?
30977             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
30978         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
30979     }
30980
30981     // public events
30982     this.addEvents({
30983         /**
30984          * @event beforeresize
30985          * Fired before resize is allowed. Set enabled to false to cancel resize.
30986          * @param {Roo.Resizable} this
30987          * @param {Roo.EventObject} e The mousedown event
30988          */
30989         "beforeresize" : true,
30990         /**
30991          * @event resizing
30992          * Fired a resizing.
30993          * @param {Roo.Resizable} this
30994          * @param {Number} x The new x position
30995          * @param {Number} y The new y position
30996          * @param {Number} w The new w width
30997          * @param {Number} h The new h hight
30998          * @param {Roo.EventObject} e The mouseup event
30999          */
31000         "resizing" : true,
31001         /**
31002          * @event resize
31003          * Fired after a resize.
31004          * @param {Roo.Resizable} this
31005          * @param {Number} width The new width
31006          * @param {Number} height The new height
31007          * @param {Roo.EventObject} e The mouseup event
31008          */
31009         "resize" : true
31010     });
31011
31012     if(this.width !== null && this.height !== null){
31013         this.resizeTo(this.width, this.height);
31014     }else{
31015         this.updateChildSize();
31016     }
31017     if(Roo.isIE){
31018         this.el.dom.style.zoom = 1;
31019     }
31020     Roo.Resizable.superclass.constructor.call(this);
31021 };
31022
31023 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31024         resizeChild : false,
31025         adjustments : [0, 0],
31026         minWidth : 5,
31027         minHeight : 5,
31028         maxWidth : 10000,
31029         maxHeight : 10000,
31030         enabled : true,
31031         animate : false,
31032         duration : .35,
31033         dynamic : false,
31034         handles : false,
31035         multiDirectional : false,
31036         disableTrackOver : false,
31037         easing : 'easeOutStrong',
31038         widthIncrement : 0,
31039         heightIncrement : 0,
31040         pinned : false,
31041         width : null,
31042         height : null,
31043         preserveRatio : false,
31044         transparent: false,
31045         minX: 0,
31046         minY: 0,
31047         draggable: false,
31048
31049         /**
31050          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31051          */
31052         constrainTo: undefined,
31053         /**
31054          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31055          */
31056         resizeRegion: undefined,
31057
31058
31059     /**
31060      * Perform a manual resize
31061      * @param {Number} width
31062      * @param {Number} height
31063      */
31064     resizeTo : function(width, height){
31065         this.el.setSize(width, height);
31066         this.updateChildSize();
31067         this.fireEvent("resize", this, width, height, null);
31068     },
31069
31070     // private
31071     startSizing : function(e, handle){
31072         this.fireEvent("beforeresize", this, e);
31073         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31074
31075             if(!this.overlay){
31076                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31077                 this.overlay.unselectable();
31078                 this.overlay.enableDisplayMode("block");
31079                 this.overlay.on("mousemove", this.onMouseMove, this);
31080                 this.overlay.on("mouseup", this.onMouseUp, this);
31081             }
31082             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31083
31084             this.resizing = true;
31085             this.startBox = this.el.getBox();
31086             this.startPoint = e.getXY();
31087             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31088                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31089
31090             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31091             this.overlay.show();
31092
31093             if(this.constrainTo) {
31094                 var ct = Roo.get(this.constrainTo);
31095                 this.resizeRegion = ct.getRegion().adjust(
31096                     ct.getFrameWidth('t'),
31097                     ct.getFrameWidth('l'),
31098                     -ct.getFrameWidth('b'),
31099                     -ct.getFrameWidth('r')
31100                 );
31101             }
31102
31103             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31104             this.proxy.show();
31105             this.proxy.setBox(this.startBox);
31106             if(!this.dynamic){
31107                 this.proxy.setStyle('visibility', 'visible');
31108             }
31109         }
31110     },
31111
31112     // private
31113     onMouseDown : function(handle, e){
31114         if(this.enabled){
31115             e.stopEvent();
31116             this.activeHandle = handle;
31117             this.startSizing(e, handle);
31118         }
31119     },
31120
31121     // private
31122     onMouseUp : function(e){
31123         var size = this.resizeElement();
31124         this.resizing = false;
31125         this.handleOut();
31126         this.overlay.hide();
31127         this.proxy.hide();
31128         this.fireEvent("resize", this, size.width, size.height, e);
31129     },
31130
31131     // private
31132     updateChildSize : function(){
31133         
31134         if(this.resizeChild){
31135             var el = this.el;
31136             var child = this.resizeChild;
31137             var adj = this.adjustments;
31138             if(el.dom.offsetWidth){
31139                 var b = el.getSize(true);
31140                 child.setSize(b.width+adj[0], b.height+adj[1]);
31141             }
31142             // Second call here for IE
31143             // The first call enables instant resizing and
31144             // the second call corrects scroll bars if they
31145             // exist
31146             if(Roo.isIE){
31147                 setTimeout(function(){
31148                     if(el.dom.offsetWidth){
31149                         var b = el.getSize(true);
31150                         child.setSize(b.width+adj[0], b.height+adj[1]);
31151                     }
31152                 }, 10);
31153             }
31154         }
31155     },
31156
31157     // private
31158     snap : function(value, inc, min){
31159         if(!inc || !value) {
31160             return value;
31161         }
31162         var newValue = value;
31163         var m = value % inc;
31164         if(m > 0){
31165             if(m > (inc/2)){
31166                 newValue = value + (inc-m);
31167             }else{
31168                 newValue = value - m;
31169             }
31170         }
31171         return Math.max(min, newValue);
31172     },
31173
31174     // private
31175     resizeElement : function(){
31176         var box = this.proxy.getBox();
31177         if(this.updateBox){
31178             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31179         }else{
31180             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31181         }
31182         this.updateChildSize();
31183         if(!this.dynamic){
31184             this.proxy.hide();
31185         }
31186         return box;
31187     },
31188
31189     // private
31190     constrain : function(v, diff, m, mx){
31191         if(v - diff < m){
31192             diff = v - m;
31193         }else if(v - diff > mx){
31194             diff = mx - v;
31195         }
31196         return diff;
31197     },
31198
31199     // private
31200     onMouseMove : function(e){
31201         
31202         if(this.enabled){
31203             try{// try catch so if something goes wrong the user doesn't get hung
31204
31205             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31206                 return;
31207             }
31208
31209             //var curXY = this.startPoint;
31210             var curSize = this.curSize || this.startBox;
31211             var x = this.startBox.x, y = this.startBox.y;
31212             var ox = x, oy = y;
31213             var w = curSize.width, h = curSize.height;
31214             var ow = w, oh = h;
31215             var mw = this.minWidth, mh = this.minHeight;
31216             var mxw = this.maxWidth, mxh = this.maxHeight;
31217             var wi = this.widthIncrement;
31218             var hi = this.heightIncrement;
31219
31220             var eventXY = e.getXY();
31221             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31222             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31223
31224             var pos = this.activeHandle.position;
31225
31226             switch(pos){
31227                 case "east":
31228                     w += diffX;
31229                     w = Math.min(Math.max(mw, w), mxw);
31230                     break;
31231              
31232                 case "south":
31233                     h += diffY;
31234                     h = Math.min(Math.max(mh, h), mxh);
31235                     break;
31236                 case "southeast":
31237                     w += diffX;
31238                     h += diffY;
31239                     w = Math.min(Math.max(mw, w), mxw);
31240                     h = Math.min(Math.max(mh, h), mxh);
31241                     break;
31242                 case "north":
31243                     diffY = this.constrain(h, diffY, mh, mxh);
31244                     y += diffY;
31245                     h -= diffY;
31246                     break;
31247                 case "hdrag":
31248                     
31249                     if (wi) {
31250                         var adiffX = Math.abs(diffX);
31251                         var sub = (adiffX % wi); // how much 
31252                         if (sub > (wi/2)) { // far enough to snap
31253                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31254                         } else {
31255                             // remove difference.. 
31256                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31257                         }
31258                     }
31259                     x += diffX;
31260                     x = Math.max(this.minX, x);
31261                     break;
31262                 case "west":
31263                     diffX = this.constrain(w, diffX, mw, mxw);
31264                     x += diffX;
31265                     w -= diffX;
31266                     break;
31267                 case "northeast":
31268                     w += diffX;
31269                     w = Math.min(Math.max(mw, w), mxw);
31270                     diffY = this.constrain(h, diffY, mh, mxh);
31271                     y += diffY;
31272                     h -= diffY;
31273                     break;
31274                 case "northwest":
31275                     diffX = this.constrain(w, diffX, mw, mxw);
31276                     diffY = this.constrain(h, diffY, mh, mxh);
31277                     y += diffY;
31278                     h -= diffY;
31279                     x += diffX;
31280                     w -= diffX;
31281                     break;
31282                case "southwest":
31283                     diffX = this.constrain(w, diffX, mw, mxw);
31284                     h += diffY;
31285                     h = Math.min(Math.max(mh, h), mxh);
31286                     x += diffX;
31287                     w -= diffX;
31288                     break;
31289             }
31290
31291             var sw = this.snap(w, wi, mw);
31292             var sh = this.snap(h, hi, mh);
31293             if(sw != w || sh != h){
31294                 switch(pos){
31295                     case "northeast":
31296                         y -= sh - h;
31297                     break;
31298                     case "north":
31299                         y -= sh - h;
31300                         break;
31301                     case "southwest":
31302                         x -= sw - w;
31303                     break;
31304                     case "west":
31305                         x -= sw - w;
31306                         break;
31307                     case "northwest":
31308                         x -= sw - w;
31309                         y -= sh - h;
31310                     break;
31311                 }
31312                 w = sw;
31313                 h = sh;
31314             }
31315
31316             if(this.preserveRatio){
31317                 switch(pos){
31318                     case "southeast":
31319                     case "east":
31320                         h = oh * (w/ow);
31321                         h = Math.min(Math.max(mh, h), mxh);
31322                         w = ow * (h/oh);
31323                        break;
31324                     case "south":
31325                         w = ow * (h/oh);
31326                         w = Math.min(Math.max(mw, w), mxw);
31327                         h = oh * (w/ow);
31328                         break;
31329                     case "northeast":
31330                         w = ow * (h/oh);
31331                         w = Math.min(Math.max(mw, w), mxw);
31332                         h = oh * (w/ow);
31333                     break;
31334                     case "north":
31335                         var tw = w;
31336                         w = ow * (h/oh);
31337                         w = Math.min(Math.max(mw, w), mxw);
31338                         h = oh * (w/ow);
31339                         x += (tw - w) / 2;
31340                         break;
31341                     case "southwest":
31342                         h = oh * (w/ow);
31343                         h = Math.min(Math.max(mh, h), mxh);
31344                         var tw = w;
31345                         w = ow * (h/oh);
31346                         x += tw - w;
31347                         break;
31348                     case "west":
31349                         var th = h;
31350                         h = oh * (w/ow);
31351                         h = Math.min(Math.max(mh, h), mxh);
31352                         y += (th - h) / 2;
31353                         var tw = w;
31354                         w = ow * (h/oh);
31355                         x += tw - w;
31356                        break;
31357                     case "northwest":
31358                         var tw = w;
31359                         var th = h;
31360                         h = oh * (w/ow);
31361                         h = Math.min(Math.max(mh, h), mxh);
31362                         w = ow * (h/oh);
31363                         y += th - h;
31364                         x += tw - w;
31365                        break;
31366
31367                 }
31368             }
31369             if (pos == 'hdrag') {
31370                 w = ow;
31371             }
31372             this.proxy.setBounds(x, y, w, h);
31373             if(this.dynamic){
31374                 this.resizeElement();
31375             }
31376             }catch(e){}
31377         }
31378         this.fireEvent("resizing", this, x, y, w, h, e);
31379     },
31380
31381     // private
31382     handleOver : function(){
31383         if(this.enabled){
31384             this.el.addClass("x-resizable-over");
31385         }
31386     },
31387
31388     // private
31389     handleOut : function(){
31390         if(!this.resizing){
31391             this.el.removeClass("x-resizable-over");
31392         }
31393     },
31394
31395     /**
31396      * Returns the element this component is bound to.
31397      * @return {Roo.Element}
31398      */
31399     getEl : function(){
31400         return this.el;
31401     },
31402
31403     /**
31404      * Returns the resizeChild element (or null).
31405      * @return {Roo.Element}
31406      */
31407     getResizeChild : function(){
31408         return this.resizeChild;
31409     },
31410     groupHandler : function()
31411     {
31412         
31413     },
31414     /**
31415      * Destroys this resizable. If the element was wrapped and
31416      * removeEl is not true then the element remains.
31417      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31418      */
31419     destroy : function(removeEl){
31420         this.proxy.remove();
31421         if(this.overlay){
31422             this.overlay.removeAllListeners();
31423             this.overlay.remove();
31424         }
31425         var ps = Roo.Resizable.positions;
31426         for(var k in ps){
31427             if(typeof ps[k] != "function" && this[ps[k]]){
31428                 var h = this[ps[k]];
31429                 h.el.removeAllListeners();
31430                 h.el.remove();
31431             }
31432         }
31433         if(removeEl){
31434             this.el.update("");
31435             this.el.remove();
31436         }
31437     }
31438 });
31439
31440 // private
31441 // hash to map config positions to true positions
31442 Roo.Resizable.positions = {
31443     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31444     hd: "hdrag"
31445 };
31446
31447 // private
31448 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31449     if(!this.tpl){
31450         // only initialize the template if resizable is used
31451         var tpl = Roo.DomHelper.createTemplate(
31452             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31453         );
31454         tpl.compile();
31455         Roo.Resizable.Handle.prototype.tpl = tpl;
31456     }
31457     this.position = pos;
31458     this.rz = rz;
31459     // show north drag fro topdra
31460     var handlepos = pos == 'hdrag' ? 'north' : pos;
31461     
31462     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31463     if (pos == 'hdrag') {
31464         this.el.setStyle('cursor', 'pointer');
31465     }
31466     this.el.unselectable();
31467     if(transparent){
31468         this.el.setOpacity(0);
31469     }
31470     this.el.on("mousedown", this.onMouseDown, this);
31471     if(!disableTrackOver){
31472         this.el.on("mouseover", this.onMouseOver, this);
31473         this.el.on("mouseout", this.onMouseOut, this);
31474     }
31475 };
31476
31477 // private
31478 Roo.Resizable.Handle.prototype = {
31479     afterResize : function(rz){
31480         Roo.log('after?');
31481         // do nothing
31482     },
31483     // private
31484     onMouseDown : function(e){
31485         this.rz.onMouseDown(this, e);
31486     },
31487     // private
31488     onMouseOver : function(e){
31489         this.rz.handleOver(this, e);
31490     },
31491     // private
31492     onMouseOut : function(e){
31493         this.rz.handleOut(this, e);
31494     }
31495 };/*
31496  * Based on:
31497  * Ext JS Library 1.1.1
31498  * Copyright(c) 2006-2007, Ext JS, LLC.
31499  *
31500  * Originally Released Under LGPL - original licence link has changed is not relivant.
31501  *
31502  * Fork - LGPL
31503  * <script type="text/javascript">
31504  */
31505
31506 /**
31507  * @class Roo.Editor
31508  * @extends Roo.Component
31509  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31510  * @constructor
31511  * Create a new Editor
31512  * @param {Roo.form.Field} field The Field object (or descendant)
31513  * @param {Object} config The config object
31514  */
31515 Roo.Editor = function(field, config){
31516     Roo.Editor.superclass.constructor.call(this, config);
31517     this.field = field;
31518     this.addEvents({
31519         /**
31520              * @event beforestartedit
31521              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31522              * false from the handler of this event.
31523              * @param {Editor} this
31524              * @param {Roo.Element} boundEl The underlying element bound to this editor
31525              * @param {Mixed} value The field value being set
31526              */
31527         "beforestartedit" : true,
31528         /**
31529              * @event startedit
31530              * Fires when this editor is displayed
31531              * @param {Roo.Element} boundEl The underlying element bound to this editor
31532              * @param {Mixed} value The starting field value
31533              */
31534         "startedit" : true,
31535         /**
31536              * @event beforecomplete
31537              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31538              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31539              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31540              * event will not fire since no edit actually occurred.
31541              * @param {Editor} this
31542              * @param {Mixed} value The current field value
31543              * @param {Mixed} startValue The original field value
31544              */
31545         "beforecomplete" : true,
31546         /**
31547              * @event complete
31548              * Fires after editing is complete and any changed value has been written to the underlying field.
31549              * @param {Editor} this
31550              * @param {Mixed} value The current field value
31551              * @param {Mixed} startValue The original field value
31552              */
31553         "complete" : true,
31554         /**
31555          * @event specialkey
31556          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31557          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31558          * @param {Roo.form.Field} this
31559          * @param {Roo.EventObject} e The event object
31560          */
31561         "specialkey" : true
31562     });
31563 };
31564
31565 Roo.extend(Roo.Editor, Roo.Component, {
31566     /**
31567      * @cfg {Boolean/String} autosize
31568      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31569      * or "height" to adopt the height only (defaults to false)
31570      */
31571     /**
31572      * @cfg {Boolean} revertInvalid
31573      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31574      * validation fails (defaults to true)
31575      */
31576     /**
31577      * @cfg {Boolean} ignoreNoChange
31578      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31579      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31580      * will never be ignored.
31581      */
31582     /**
31583      * @cfg {Boolean} hideEl
31584      * False to keep the bound element visible while the editor is displayed (defaults to true)
31585      */
31586     /**
31587      * @cfg {Mixed} value
31588      * The data value of the underlying field (defaults to "")
31589      */
31590     value : "",
31591     /**
31592      * @cfg {String} alignment
31593      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31594      */
31595     alignment: "c-c?",
31596     /**
31597      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31598      * for bottom-right shadow (defaults to "frame")
31599      */
31600     shadow : "frame",
31601     /**
31602      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31603      */
31604     constrain : false,
31605     /**
31606      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31607      */
31608     completeOnEnter : false,
31609     /**
31610      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31611      */
31612     cancelOnEsc : false,
31613     /**
31614      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31615      */
31616     updateEl : false,
31617
31618     // private
31619     onRender : function(ct, position){
31620         this.el = new Roo.Layer({
31621             shadow: this.shadow,
31622             cls: "x-editor",
31623             parentEl : ct,
31624             shim : this.shim,
31625             shadowOffset:4,
31626             id: this.id,
31627             constrain: this.constrain
31628         });
31629         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31630         if(this.field.msgTarget != 'title'){
31631             this.field.msgTarget = 'qtip';
31632         }
31633         this.field.render(this.el);
31634         if(Roo.isGecko){
31635             this.field.el.dom.setAttribute('autocomplete', 'off');
31636         }
31637         this.field.on("specialkey", this.onSpecialKey, this);
31638         if(this.swallowKeys){
31639             this.field.el.swallowEvent(['keydown','keypress']);
31640         }
31641         this.field.show();
31642         this.field.on("blur", this.onBlur, this);
31643         if(this.field.grow){
31644             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31645         }
31646     },
31647
31648     onSpecialKey : function(field, e)
31649     {
31650         //Roo.log('editor onSpecialKey');
31651         if(this.completeOnEnter && e.getKey() == e.ENTER){
31652             e.stopEvent();
31653             this.completeEdit();
31654             return;
31655         }
31656         // do not fire special key otherwise it might hide close the editor...
31657         if(e.getKey() == e.ENTER){    
31658             return;
31659         }
31660         if(this.cancelOnEsc && e.getKey() == e.ESC){
31661             this.cancelEdit();
31662             return;
31663         } 
31664         this.fireEvent('specialkey', field, e);
31665     
31666     },
31667
31668     /**
31669      * Starts the editing process and shows the editor.
31670      * @param {String/HTMLElement/Element} el The element to edit
31671      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31672       * to the innerHTML of el.
31673      */
31674     startEdit : function(el, value){
31675         if(this.editing){
31676             this.completeEdit();
31677         }
31678         this.boundEl = Roo.get(el);
31679         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31680         if(!this.rendered){
31681             this.render(this.parentEl || document.body);
31682         }
31683         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31684             return;
31685         }
31686         this.startValue = v;
31687         this.field.setValue(v);
31688         if(this.autoSize){
31689             var sz = this.boundEl.getSize();
31690             switch(this.autoSize){
31691                 case "width":
31692                 this.setSize(sz.width,  "");
31693                 break;
31694                 case "height":
31695                 this.setSize("",  sz.height);
31696                 break;
31697                 default:
31698                 this.setSize(sz.width,  sz.height);
31699             }
31700         }
31701         this.el.alignTo(this.boundEl, this.alignment);
31702         this.editing = true;
31703         if(Roo.QuickTips){
31704             Roo.QuickTips.disable();
31705         }
31706         this.show();
31707     },
31708
31709     /**
31710      * Sets the height and width of this editor.
31711      * @param {Number} width The new width
31712      * @param {Number} height The new height
31713      */
31714     setSize : function(w, h){
31715         this.field.setSize(w, h);
31716         if(this.el){
31717             this.el.sync();
31718         }
31719     },
31720
31721     /**
31722      * Realigns the editor to the bound field based on the current alignment config value.
31723      */
31724     realign : function(){
31725         this.el.alignTo(this.boundEl, this.alignment);
31726     },
31727
31728     /**
31729      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31730      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31731      */
31732     completeEdit : function(remainVisible){
31733         if(!this.editing){
31734             return;
31735         }
31736         var v = this.getValue();
31737         if(this.revertInvalid !== false && !this.field.isValid()){
31738             v = this.startValue;
31739             this.cancelEdit(true);
31740         }
31741         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31742             this.editing = false;
31743             this.hide();
31744             return;
31745         }
31746         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31747             this.editing = false;
31748             if(this.updateEl && this.boundEl){
31749                 this.boundEl.update(v);
31750             }
31751             if(remainVisible !== true){
31752                 this.hide();
31753             }
31754             this.fireEvent("complete", this, v, this.startValue);
31755         }
31756     },
31757
31758     // private
31759     onShow : function(){
31760         this.el.show();
31761         if(this.hideEl !== false){
31762             this.boundEl.hide();
31763         }
31764         this.field.show();
31765         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31766             this.fixIEFocus = true;
31767             this.deferredFocus.defer(50, this);
31768         }else{
31769             this.field.focus();
31770         }
31771         this.fireEvent("startedit", this.boundEl, this.startValue);
31772     },
31773
31774     deferredFocus : function(){
31775         if(this.editing){
31776             this.field.focus();
31777         }
31778     },
31779
31780     /**
31781      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31782      * reverted to the original starting value.
31783      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31784      * cancel (defaults to false)
31785      */
31786     cancelEdit : function(remainVisible){
31787         if(this.editing){
31788             this.setValue(this.startValue);
31789             if(remainVisible !== true){
31790                 this.hide();
31791             }
31792         }
31793     },
31794
31795     // private
31796     onBlur : function(){
31797         if(this.allowBlur !== true && this.editing){
31798             this.completeEdit();
31799         }
31800     },
31801
31802     // private
31803     onHide : function(){
31804         if(this.editing){
31805             this.completeEdit();
31806             return;
31807         }
31808         this.field.blur();
31809         if(this.field.collapse){
31810             this.field.collapse();
31811         }
31812         this.el.hide();
31813         if(this.hideEl !== false){
31814             this.boundEl.show();
31815         }
31816         if(Roo.QuickTips){
31817             Roo.QuickTips.enable();
31818         }
31819     },
31820
31821     /**
31822      * Sets the data value of the editor
31823      * @param {Mixed} value Any valid value supported by the underlying field
31824      */
31825     setValue : function(v){
31826         this.field.setValue(v);
31827     },
31828
31829     /**
31830      * Gets the data value of the editor
31831      * @return {Mixed} The data value
31832      */
31833     getValue : function(){
31834         return this.field.getValue();
31835     }
31836 });/*
31837  * Based on:
31838  * Ext JS Library 1.1.1
31839  * Copyright(c) 2006-2007, Ext JS, LLC.
31840  *
31841  * Originally Released Under LGPL - original licence link has changed is not relivant.
31842  *
31843  * Fork - LGPL
31844  * <script type="text/javascript">
31845  */
31846  
31847 /**
31848  * @class Roo.BasicDialog
31849  * @extends Roo.util.Observable
31850  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31851  * <pre><code>
31852 var dlg = new Roo.BasicDialog("my-dlg", {
31853     height: 200,
31854     width: 300,
31855     minHeight: 100,
31856     minWidth: 150,
31857     modal: true,
31858     proxyDrag: true,
31859     shadow: true
31860 });
31861 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31862 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31863 dlg.addButton('Cancel', dlg.hide, dlg);
31864 dlg.show();
31865 </code></pre>
31866   <b>A Dialog should always be a direct child of the body element.</b>
31867  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31868  * @cfg {String} title Default text to display in the title bar (defaults to null)
31869  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31870  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31871  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31872  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31873  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31874  * (defaults to null with no animation)
31875  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31876  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31877  * property for valid values (defaults to 'all')
31878  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31879  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31880  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31881  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31882  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31883  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31884  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31885  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31886  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31887  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31888  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31889  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31890  * draggable = true (defaults to false)
31891  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31892  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31893  * shadow (defaults to false)
31894  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31895  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31896  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31897  * @cfg {Array} buttons Array of buttons
31898  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31899  * @constructor
31900  * Create a new BasicDialog.
31901  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31902  * @param {Object} config Configuration options
31903  */
31904 Roo.BasicDialog = function(el, config){
31905     this.el = Roo.get(el);
31906     var dh = Roo.DomHelper;
31907     if(!this.el && config && config.autoCreate){
31908         if(typeof config.autoCreate == "object"){
31909             if(!config.autoCreate.id){
31910                 config.autoCreate.id = el;
31911             }
31912             this.el = dh.append(document.body,
31913                         config.autoCreate, true);
31914         }else{
31915             this.el = dh.append(document.body,
31916                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31917         }
31918     }
31919     el = this.el;
31920     el.setDisplayed(true);
31921     el.hide = this.hideAction;
31922     this.id = el.id;
31923     el.addClass("x-dlg");
31924
31925     Roo.apply(this, config);
31926
31927     this.proxy = el.createProxy("x-dlg-proxy");
31928     this.proxy.hide = this.hideAction;
31929     this.proxy.setOpacity(.5);
31930     this.proxy.hide();
31931
31932     if(config.width){
31933         el.setWidth(config.width);
31934     }
31935     if(config.height){
31936         el.setHeight(config.height);
31937     }
31938     this.size = el.getSize();
31939     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31940         this.xy = [config.x,config.y];
31941     }else{
31942         this.xy = el.getCenterXY(true);
31943     }
31944     /** The header element @type Roo.Element */
31945     this.header = el.child("> .x-dlg-hd");
31946     /** The body element @type Roo.Element */
31947     this.body = el.child("> .x-dlg-bd");
31948     /** The footer element @type Roo.Element */
31949     this.footer = el.child("> .x-dlg-ft");
31950
31951     if(!this.header){
31952         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31953     }
31954     if(!this.body){
31955         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31956     }
31957
31958     this.header.unselectable();
31959     if(this.title){
31960         this.header.update(this.title);
31961     }
31962     // this element allows the dialog to be focused for keyboard event
31963     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
31964     this.focusEl.swallowEvent("click", true);
31965
31966     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
31967
31968     // wrap the body and footer for special rendering
31969     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
31970     if(this.footer){
31971         this.bwrap.dom.appendChild(this.footer.dom);
31972     }
31973
31974     this.bg = this.el.createChild({
31975         tag: "div", cls:"x-dlg-bg",
31976         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
31977     });
31978     this.centerBg = this.bg.child("div.x-dlg-bg-center");
31979
31980
31981     if(this.autoScroll !== false && !this.autoTabs){
31982         this.body.setStyle("overflow", "auto");
31983     }
31984
31985     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
31986
31987     if(this.closable !== false){
31988         this.el.addClass("x-dlg-closable");
31989         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
31990         this.close.on("click", this.closeClick, this);
31991         this.close.addClassOnOver("x-dlg-close-over");
31992     }
31993     if(this.collapsible !== false){
31994         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
31995         this.collapseBtn.on("click", this.collapseClick, this);
31996         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
31997         this.header.on("dblclick", this.collapseClick, this);
31998     }
31999     if(this.resizable !== false){
32000         this.el.addClass("x-dlg-resizable");
32001         this.resizer = new Roo.Resizable(el, {
32002             minWidth: this.minWidth || 80,
32003             minHeight:this.minHeight || 80,
32004             handles: this.resizeHandles || "all",
32005             pinned: true
32006         });
32007         this.resizer.on("beforeresize", this.beforeResize, this);
32008         this.resizer.on("resize", this.onResize, this);
32009     }
32010     if(this.draggable !== false){
32011         el.addClass("x-dlg-draggable");
32012         if (!this.proxyDrag) {
32013             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32014         }
32015         else {
32016             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32017         }
32018         dd.setHandleElId(this.header.id);
32019         dd.endDrag = this.endMove.createDelegate(this);
32020         dd.startDrag = this.startMove.createDelegate(this);
32021         dd.onDrag = this.onDrag.createDelegate(this);
32022         dd.scroll = false;
32023         this.dd = dd;
32024     }
32025     if(this.modal){
32026         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32027         this.mask.enableDisplayMode("block");
32028         this.mask.hide();
32029         this.el.addClass("x-dlg-modal");
32030     }
32031     if(this.shadow){
32032         this.shadow = new Roo.Shadow({
32033             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32034             offset : this.shadowOffset
32035         });
32036     }else{
32037         this.shadowOffset = 0;
32038     }
32039     if(Roo.useShims && this.shim !== false){
32040         this.shim = this.el.createShim();
32041         this.shim.hide = this.hideAction;
32042         this.shim.hide();
32043     }else{
32044         this.shim = false;
32045     }
32046     if(this.autoTabs){
32047         this.initTabs();
32048     }
32049     if (this.buttons) { 
32050         var bts= this.buttons;
32051         this.buttons = [];
32052         Roo.each(bts, function(b) {
32053             this.addButton(b);
32054         }, this);
32055     }
32056     
32057     
32058     this.addEvents({
32059         /**
32060          * @event keydown
32061          * Fires when a key is pressed
32062          * @param {Roo.BasicDialog} this
32063          * @param {Roo.EventObject} e
32064          */
32065         "keydown" : true,
32066         /**
32067          * @event move
32068          * Fires when this dialog is moved by the user.
32069          * @param {Roo.BasicDialog} this
32070          * @param {Number} x The new page X
32071          * @param {Number} y The new page Y
32072          */
32073         "move" : true,
32074         /**
32075          * @event resize
32076          * Fires when this dialog is resized by the user.
32077          * @param {Roo.BasicDialog} this
32078          * @param {Number} width The new width
32079          * @param {Number} height The new height
32080          */
32081         "resize" : true,
32082         /**
32083          * @event beforehide
32084          * Fires before this dialog is hidden.
32085          * @param {Roo.BasicDialog} this
32086          */
32087         "beforehide" : true,
32088         /**
32089          * @event hide
32090          * Fires when this dialog is hidden.
32091          * @param {Roo.BasicDialog} this
32092          */
32093         "hide" : true,
32094         /**
32095          * @event beforeshow
32096          * Fires before this dialog is shown.
32097          * @param {Roo.BasicDialog} this
32098          */
32099         "beforeshow" : true,
32100         /**
32101          * @event show
32102          * Fires when this dialog is shown.
32103          * @param {Roo.BasicDialog} this
32104          */
32105         "show" : true
32106     });
32107     el.on("keydown", this.onKeyDown, this);
32108     el.on("mousedown", this.toFront, this);
32109     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32110     this.el.hide();
32111     Roo.DialogManager.register(this);
32112     Roo.BasicDialog.superclass.constructor.call(this);
32113 };
32114
32115 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32116     shadowOffset: Roo.isIE ? 6 : 5,
32117     minHeight: 80,
32118     minWidth: 200,
32119     minButtonWidth: 75,
32120     defaultButton: null,
32121     buttonAlign: "right",
32122     tabTag: 'div',
32123     firstShow: true,
32124
32125     /**
32126      * Sets the dialog title text
32127      * @param {String} text The title text to display
32128      * @return {Roo.BasicDialog} this
32129      */
32130     setTitle : function(text){
32131         this.header.update(text);
32132         return this;
32133     },
32134
32135     // private
32136     closeClick : function(){
32137         this.hide();
32138     },
32139
32140     // private
32141     collapseClick : function(){
32142         this[this.collapsed ? "expand" : "collapse"]();
32143     },
32144
32145     /**
32146      * Collapses the dialog to its minimized state (only the title bar is visible).
32147      * Equivalent to the user clicking the collapse dialog button.
32148      */
32149     collapse : function(){
32150         if(!this.collapsed){
32151             this.collapsed = true;
32152             this.el.addClass("x-dlg-collapsed");
32153             this.restoreHeight = this.el.getHeight();
32154             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32155         }
32156     },
32157
32158     /**
32159      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32160      * clicking the expand dialog button.
32161      */
32162     expand : function(){
32163         if(this.collapsed){
32164             this.collapsed = false;
32165             this.el.removeClass("x-dlg-collapsed");
32166             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32167         }
32168     },
32169
32170     /**
32171      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32172      * @return {Roo.TabPanel} The tabs component
32173      */
32174     initTabs : function(){
32175         var tabs = this.getTabs();
32176         while(tabs.getTab(0)){
32177             tabs.removeTab(0);
32178         }
32179         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32180             var dom = el.dom;
32181             tabs.addTab(Roo.id(dom), dom.title);
32182             dom.title = "";
32183         });
32184         tabs.activate(0);
32185         return tabs;
32186     },
32187
32188     // private
32189     beforeResize : function(){
32190         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32191     },
32192
32193     // private
32194     onResize : function(){
32195         this.refreshSize();
32196         this.syncBodyHeight();
32197         this.adjustAssets();
32198         this.focus();
32199         this.fireEvent("resize", this, this.size.width, this.size.height);
32200     },
32201
32202     // private
32203     onKeyDown : function(e){
32204         if(this.isVisible()){
32205             this.fireEvent("keydown", this, e);
32206         }
32207     },
32208
32209     /**
32210      * Resizes the dialog.
32211      * @param {Number} width
32212      * @param {Number} height
32213      * @return {Roo.BasicDialog} this
32214      */
32215     resizeTo : function(width, height){
32216         this.el.setSize(width, height);
32217         this.size = {width: width, height: height};
32218         this.syncBodyHeight();
32219         if(this.fixedcenter){
32220             this.center();
32221         }
32222         if(this.isVisible()){
32223             this.constrainXY();
32224             this.adjustAssets();
32225         }
32226         this.fireEvent("resize", this, width, height);
32227         return this;
32228     },
32229
32230
32231     /**
32232      * Resizes the dialog to fit the specified content size.
32233      * @param {Number} width
32234      * @param {Number} height
32235      * @return {Roo.BasicDialog} this
32236      */
32237     setContentSize : function(w, h){
32238         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32239         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32240         //if(!this.el.isBorderBox()){
32241             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32242             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32243         //}
32244         if(this.tabs){
32245             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32246             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32247         }
32248         this.resizeTo(w, h);
32249         return this;
32250     },
32251
32252     /**
32253      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32254      * executed in response to a particular key being pressed while the dialog is active.
32255      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32256      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32257      * @param {Function} fn The function to call
32258      * @param {Object} scope (optional) The scope of the function
32259      * @return {Roo.BasicDialog} this
32260      */
32261     addKeyListener : function(key, fn, scope){
32262         var keyCode, shift, ctrl, alt;
32263         if(typeof key == "object" && !(key instanceof Array)){
32264             keyCode = key["key"];
32265             shift = key["shift"];
32266             ctrl = key["ctrl"];
32267             alt = key["alt"];
32268         }else{
32269             keyCode = key;
32270         }
32271         var handler = function(dlg, e){
32272             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32273                 var k = e.getKey();
32274                 if(keyCode instanceof Array){
32275                     for(var i = 0, len = keyCode.length; i < len; i++){
32276                         if(keyCode[i] == k){
32277                           fn.call(scope || window, dlg, k, e);
32278                           return;
32279                         }
32280                     }
32281                 }else{
32282                     if(k == keyCode){
32283                         fn.call(scope || window, dlg, k, e);
32284                     }
32285                 }
32286             }
32287         };
32288         this.on("keydown", handler);
32289         return this;
32290     },
32291
32292     /**
32293      * Returns the TabPanel component (creates it if it doesn't exist).
32294      * Note: If you wish to simply check for the existence of tabs without creating them,
32295      * check for a null 'tabs' property.
32296      * @return {Roo.TabPanel} The tabs component
32297      */
32298     getTabs : function(){
32299         if(!this.tabs){
32300             this.el.addClass("x-dlg-auto-tabs");
32301             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32302             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32303         }
32304         return this.tabs;
32305     },
32306
32307     /**
32308      * Adds a button to the footer section of the dialog.
32309      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32310      * object or a valid Roo.DomHelper element config
32311      * @param {Function} handler The function called when the button is clicked
32312      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32313      * @return {Roo.Button} The new button
32314      */
32315     addButton : function(config, handler, scope){
32316         var dh = Roo.DomHelper;
32317         if(!this.footer){
32318             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32319         }
32320         if(!this.btnContainer){
32321             var tb = this.footer.createChild({
32322
32323                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32324                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32325             }, null, true);
32326             this.btnContainer = tb.firstChild.firstChild.firstChild;
32327         }
32328         var bconfig = {
32329             handler: handler,
32330             scope: scope,
32331             minWidth: this.minButtonWidth,
32332             hideParent:true
32333         };
32334         if(typeof config == "string"){
32335             bconfig.text = config;
32336         }else{
32337             if(config.tag){
32338                 bconfig.dhconfig = config;
32339             }else{
32340                 Roo.apply(bconfig, config);
32341             }
32342         }
32343         var fc = false;
32344         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32345             bconfig.position = Math.max(0, bconfig.position);
32346             fc = this.btnContainer.childNodes[bconfig.position];
32347         }
32348          
32349         var btn = new Roo.Button(
32350             fc ? 
32351                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32352                 : this.btnContainer.appendChild(document.createElement("td")),
32353             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32354             bconfig
32355         );
32356         this.syncBodyHeight();
32357         if(!this.buttons){
32358             /**
32359              * Array of all the buttons that have been added to this dialog via addButton
32360              * @type Array
32361              */
32362             this.buttons = [];
32363         }
32364         this.buttons.push(btn);
32365         return btn;
32366     },
32367
32368     /**
32369      * Sets the default button to be focused when the dialog is displayed.
32370      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32371      * @return {Roo.BasicDialog} this
32372      */
32373     setDefaultButton : function(btn){
32374         this.defaultButton = btn;
32375         return this;
32376     },
32377
32378     // private
32379     getHeaderFooterHeight : function(safe){
32380         var height = 0;
32381         if(this.header){
32382            height += this.header.getHeight();
32383         }
32384         if(this.footer){
32385            var fm = this.footer.getMargins();
32386             height += (this.footer.getHeight()+fm.top+fm.bottom);
32387         }
32388         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32389         height += this.centerBg.getPadding("tb");
32390         return height;
32391     },
32392
32393     // private
32394     syncBodyHeight : function()
32395     {
32396         var bd = this.body, // the text
32397             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32398             bw = this.bwrap;
32399         var height = this.size.height - this.getHeaderFooterHeight(false);
32400         bd.setHeight(height-bd.getMargins("tb"));
32401         var hh = this.header.getHeight();
32402         var h = this.size.height-hh;
32403         cb.setHeight(h);
32404         
32405         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32406         bw.setHeight(h-cb.getPadding("tb"));
32407         
32408         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32409         bd.setWidth(bw.getWidth(true));
32410         if(this.tabs){
32411             this.tabs.syncHeight();
32412             if(Roo.isIE){
32413                 this.tabs.el.repaint();
32414             }
32415         }
32416     },
32417
32418     /**
32419      * Restores the previous state of the dialog if Roo.state is configured.
32420      * @return {Roo.BasicDialog} this
32421      */
32422     restoreState : function(){
32423         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32424         if(box && box.width){
32425             this.xy = [box.x, box.y];
32426             this.resizeTo(box.width, box.height);
32427         }
32428         return this;
32429     },
32430
32431     // private
32432     beforeShow : function(){
32433         this.expand();
32434         if(this.fixedcenter){
32435             this.xy = this.el.getCenterXY(true);
32436         }
32437         if(this.modal){
32438             Roo.get(document.body).addClass("x-body-masked");
32439             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32440             this.mask.show();
32441         }
32442         this.constrainXY();
32443     },
32444
32445     // private
32446     animShow : function(){
32447         var b = Roo.get(this.animateTarget).getBox();
32448         this.proxy.setSize(b.width, b.height);
32449         this.proxy.setLocation(b.x, b.y);
32450         this.proxy.show();
32451         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32452                     true, .35, this.showEl.createDelegate(this));
32453     },
32454
32455     /**
32456      * Shows the dialog.
32457      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32458      * @return {Roo.BasicDialog} this
32459      */
32460     show : function(animateTarget){
32461         if (this.fireEvent("beforeshow", this) === false){
32462             return;
32463         }
32464         if(this.syncHeightBeforeShow){
32465             this.syncBodyHeight();
32466         }else if(this.firstShow){
32467             this.firstShow = false;
32468             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32469         }
32470         this.animateTarget = animateTarget || this.animateTarget;
32471         if(!this.el.isVisible()){
32472             this.beforeShow();
32473             if(this.animateTarget && Roo.get(this.animateTarget)){
32474                 this.animShow();
32475             }else{
32476                 this.showEl();
32477             }
32478         }
32479         return this;
32480     },
32481
32482     // private
32483     showEl : function(){
32484         this.proxy.hide();
32485         this.el.setXY(this.xy);
32486         this.el.show();
32487         this.adjustAssets(true);
32488         this.toFront();
32489         this.focus();
32490         // IE peekaboo bug - fix found by Dave Fenwick
32491         if(Roo.isIE){
32492             this.el.repaint();
32493         }
32494         this.fireEvent("show", this);
32495     },
32496
32497     /**
32498      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32499      * dialog itself will receive focus.
32500      */
32501     focus : function(){
32502         if(this.defaultButton){
32503             this.defaultButton.focus();
32504         }else{
32505             this.focusEl.focus();
32506         }
32507     },
32508
32509     // private
32510     constrainXY : function(){
32511         if(this.constraintoviewport !== false){
32512             if(!this.viewSize){
32513                 if(this.container){
32514                     var s = this.container.getSize();
32515                     this.viewSize = [s.width, s.height];
32516                 }else{
32517                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32518                 }
32519             }
32520             var s = Roo.get(this.container||document).getScroll();
32521
32522             var x = this.xy[0], y = this.xy[1];
32523             var w = this.size.width, h = this.size.height;
32524             var vw = this.viewSize[0], vh = this.viewSize[1];
32525             // only move it if it needs it
32526             var moved = false;
32527             // first validate right/bottom
32528             if(x + w > vw+s.left){
32529                 x = vw - w;
32530                 moved = true;
32531             }
32532             if(y + h > vh+s.top){
32533                 y = vh - h;
32534                 moved = true;
32535             }
32536             // then make sure top/left isn't negative
32537             if(x < s.left){
32538                 x = s.left;
32539                 moved = true;
32540             }
32541             if(y < s.top){
32542                 y = s.top;
32543                 moved = true;
32544             }
32545             if(moved){
32546                 // cache xy
32547                 this.xy = [x, y];
32548                 if(this.isVisible()){
32549                     this.el.setLocation(x, y);
32550                     this.adjustAssets();
32551                 }
32552             }
32553         }
32554     },
32555
32556     // private
32557     onDrag : function(){
32558         if(!this.proxyDrag){
32559             this.xy = this.el.getXY();
32560             this.adjustAssets();
32561         }
32562     },
32563
32564     // private
32565     adjustAssets : function(doShow){
32566         var x = this.xy[0], y = this.xy[1];
32567         var w = this.size.width, h = this.size.height;
32568         if(doShow === true){
32569             if(this.shadow){
32570                 this.shadow.show(this.el);
32571             }
32572             if(this.shim){
32573                 this.shim.show();
32574             }
32575         }
32576         if(this.shadow && this.shadow.isVisible()){
32577             this.shadow.show(this.el);
32578         }
32579         if(this.shim && this.shim.isVisible()){
32580             this.shim.setBounds(x, y, w, h);
32581         }
32582     },
32583
32584     // private
32585     adjustViewport : function(w, h){
32586         if(!w || !h){
32587             w = Roo.lib.Dom.getViewWidth();
32588             h = Roo.lib.Dom.getViewHeight();
32589         }
32590         // cache the size
32591         this.viewSize = [w, h];
32592         if(this.modal && this.mask.isVisible()){
32593             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32594             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32595         }
32596         if(this.isVisible()){
32597             this.constrainXY();
32598         }
32599     },
32600
32601     /**
32602      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32603      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32604      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32605      */
32606     destroy : function(removeEl){
32607         if(this.isVisible()){
32608             this.animateTarget = null;
32609             this.hide();
32610         }
32611         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32612         if(this.tabs){
32613             this.tabs.destroy(removeEl);
32614         }
32615         Roo.destroy(
32616              this.shim,
32617              this.proxy,
32618              this.resizer,
32619              this.close,
32620              this.mask
32621         );
32622         if(this.dd){
32623             this.dd.unreg();
32624         }
32625         if(this.buttons){
32626            for(var i = 0, len = this.buttons.length; i < len; i++){
32627                this.buttons[i].destroy();
32628            }
32629         }
32630         this.el.removeAllListeners();
32631         if(removeEl === true){
32632             this.el.update("");
32633             this.el.remove();
32634         }
32635         Roo.DialogManager.unregister(this);
32636     },
32637
32638     // private
32639     startMove : function(){
32640         if(this.proxyDrag){
32641             this.proxy.show();
32642         }
32643         if(this.constraintoviewport !== false){
32644             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32645         }
32646     },
32647
32648     // private
32649     endMove : function(){
32650         if(!this.proxyDrag){
32651             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32652         }else{
32653             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32654             this.proxy.hide();
32655         }
32656         this.refreshSize();
32657         this.adjustAssets();
32658         this.focus();
32659         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32660     },
32661
32662     /**
32663      * Brings this dialog to the front of any other visible dialogs
32664      * @return {Roo.BasicDialog} this
32665      */
32666     toFront : function(){
32667         Roo.DialogManager.bringToFront(this);
32668         return this;
32669     },
32670
32671     /**
32672      * Sends this dialog to the back (under) of any other visible dialogs
32673      * @return {Roo.BasicDialog} this
32674      */
32675     toBack : function(){
32676         Roo.DialogManager.sendToBack(this);
32677         return this;
32678     },
32679
32680     /**
32681      * Centers this dialog in the viewport
32682      * @return {Roo.BasicDialog} this
32683      */
32684     center : function(){
32685         var xy = this.el.getCenterXY(true);
32686         this.moveTo(xy[0], xy[1]);
32687         return this;
32688     },
32689
32690     /**
32691      * Moves the dialog's top-left corner to the specified point
32692      * @param {Number} x
32693      * @param {Number} y
32694      * @return {Roo.BasicDialog} this
32695      */
32696     moveTo : function(x, y){
32697         this.xy = [x,y];
32698         if(this.isVisible()){
32699             this.el.setXY(this.xy);
32700             this.adjustAssets();
32701         }
32702         return this;
32703     },
32704
32705     /**
32706      * Aligns the dialog to the specified element
32707      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32708      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32709      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32710      * @return {Roo.BasicDialog} this
32711      */
32712     alignTo : function(element, position, offsets){
32713         this.xy = this.el.getAlignToXY(element, position, offsets);
32714         if(this.isVisible()){
32715             this.el.setXY(this.xy);
32716             this.adjustAssets();
32717         }
32718         return this;
32719     },
32720
32721     /**
32722      * Anchors an element to another element and realigns it when the window is resized.
32723      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32724      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32725      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32726      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32727      * is a number, it is used as the buffer delay (defaults to 50ms).
32728      * @return {Roo.BasicDialog} this
32729      */
32730     anchorTo : function(el, alignment, offsets, monitorScroll){
32731         var action = function(){
32732             this.alignTo(el, alignment, offsets);
32733         };
32734         Roo.EventManager.onWindowResize(action, this);
32735         var tm = typeof monitorScroll;
32736         if(tm != 'undefined'){
32737             Roo.EventManager.on(window, 'scroll', action, this,
32738                 {buffer: tm == 'number' ? monitorScroll : 50});
32739         }
32740         action.call(this);
32741         return this;
32742     },
32743
32744     /**
32745      * Returns true if the dialog is visible
32746      * @return {Boolean}
32747      */
32748     isVisible : function(){
32749         return this.el.isVisible();
32750     },
32751
32752     // private
32753     animHide : function(callback){
32754         var b = Roo.get(this.animateTarget).getBox();
32755         this.proxy.show();
32756         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32757         this.el.hide();
32758         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32759                     this.hideEl.createDelegate(this, [callback]));
32760     },
32761
32762     /**
32763      * Hides the dialog.
32764      * @param {Function} callback (optional) Function to call when the dialog is hidden
32765      * @return {Roo.BasicDialog} this
32766      */
32767     hide : function(callback){
32768         if (this.fireEvent("beforehide", this) === false){
32769             return;
32770         }
32771         if(this.shadow){
32772             this.shadow.hide();
32773         }
32774         if(this.shim) {
32775           this.shim.hide();
32776         }
32777         // sometimes animateTarget seems to get set.. causing problems...
32778         // this just double checks..
32779         if(this.animateTarget && Roo.get(this.animateTarget)) {
32780            this.animHide(callback);
32781         }else{
32782             this.el.hide();
32783             this.hideEl(callback);
32784         }
32785         return this;
32786     },
32787
32788     // private
32789     hideEl : function(callback){
32790         this.proxy.hide();
32791         if(this.modal){
32792             this.mask.hide();
32793             Roo.get(document.body).removeClass("x-body-masked");
32794         }
32795         this.fireEvent("hide", this);
32796         if(typeof callback == "function"){
32797             callback();
32798         }
32799     },
32800
32801     // private
32802     hideAction : function(){
32803         this.setLeft("-10000px");
32804         this.setTop("-10000px");
32805         this.setStyle("visibility", "hidden");
32806     },
32807
32808     // private
32809     refreshSize : function(){
32810         this.size = this.el.getSize();
32811         this.xy = this.el.getXY();
32812         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32813     },
32814
32815     // private
32816     // z-index is managed by the DialogManager and may be overwritten at any time
32817     setZIndex : function(index){
32818         if(this.modal){
32819             this.mask.setStyle("z-index", index);
32820         }
32821         if(this.shim){
32822             this.shim.setStyle("z-index", ++index);
32823         }
32824         if(this.shadow){
32825             this.shadow.setZIndex(++index);
32826         }
32827         this.el.setStyle("z-index", ++index);
32828         if(this.proxy){
32829             this.proxy.setStyle("z-index", ++index);
32830         }
32831         if(this.resizer){
32832             this.resizer.proxy.setStyle("z-index", ++index);
32833         }
32834
32835         this.lastZIndex = index;
32836     },
32837
32838     /**
32839      * Returns the element for this dialog
32840      * @return {Roo.Element} The underlying dialog Element
32841      */
32842     getEl : function(){
32843         return this.el;
32844     }
32845 });
32846
32847 /**
32848  * @class Roo.DialogManager
32849  * Provides global access to BasicDialogs that have been created and
32850  * support for z-indexing (layering) multiple open dialogs.
32851  */
32852 Roo.DialogManager = function(){
32853     var list = {};
32854     var accessList = [];
32855     var front = null;
32856
32857     // private
32858     var sortDialogs = function(d1, d2){
32859         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32860     };
32861
32862     // private
32863     var orderDialogs = function(){
32864         accessList.sort(sortDialogs);
32865         var seed = Roo.DialogManager.zseed;
32866         for(var i = 0, len = accessList.length; i < len; i++){
32867             var dlg = accessList[i];
32868             if(dlg){
32869                 dlg.setZIndex(seed + (i*10));
32870             }
32871         }
32872     };
32873
32874     return {
32875         /**
32876          * The starting z-index for BasicDialogs (defaults to 9000)
32877          * @type Number The z-index value
32878          */
32879         zseed : 9000,
32880
32881         // private
32882         register : function(dlg){
32883             list[dlg.id] = dlg;
32884             accessList.push(dlg);
32885         },
32886
32887         // private
32888         unregister : function(dlg){
32889             delete list[dlg.id];
32890             var i=0;
32891             var len=0;
32892             if(!accessList.indexOf){
32893                 for(  i = 0, len = accessList.length; i < len; i++){
32894                     if(accessList[i] == dlg){
32895                         accessList.splice(i, 1);
32896                         return;
32897                     }
32898                 }
32899             }else{
32900                  i = accessList.indexOf(dlg);
32901                 if(i != -1){
32902                     accessList.splice(i, 1);
32903                 }
32904             }
32905         },
32906
32907         /**
32908          * Gets a registered dialog by id
32909          * @param {String/Object} id The id of the dialog or a dialog
32910          * @return {Roo.BasicDialog} this
32911          */
32912         get : function(id){
32913             return typeof id == "object" ? id : list[id];
32914         },
32915
32916         /**
32917          * Brings the specified dialog to the front
32918          * @param {String/Object} dlg The id of the dialog or a dialog
32919          * @return {Roo.BasicDialog} this
32920          */
32921         bringToFront : function(dlg){
32922             dlg = this.get(dlg);
32923             if(dlg != front){
32924                 front = dlg;
32925                 dlg._lastAccess = new Date().getTime();
32926                 orderDialogs();
32927             }
32928             return dlg;
32929         },
32930
32931         /**
32932          * Sends the specified dialog to the back
32933          * @param {String/Object} dlg The id of the dialog or a dialog
32934          * @return {Roo.BasicDialog} this
32935          */
32936         sendToBack : function(dlg){
32937             dlg = this.get(dlg);
32938             dlg._lastAccess = -(new Date().getTime());
32939             orderDialogs();
32940             return dlg;
32941         },
32942
32943         /**
32944          * Hides all dialogs
32945          */
32946         hideAll : function(){
32947             for(var id in list){
32948                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32949                     list[id].hide();
32950                 }
32951             }
32952         }
32953     };
32954 }();
32955
32956 /**
32957  * @class Roo.LayoutDialog
32958  * @extends Roo.BasicDialog
32959  * Dialog which provides adjustments for working with a layout in a Dialog.
32960  * Add your necessary layout config options to the dialog's config.<br>
32961  * Example usage (including a nested layout):
32962  * <pre><code>
32963 if(!dialog){
32964     dialog = new Roo.LayoutDialog("download-dlg", {
32965         modal: true,
32966         width:600,
32967         height:450,
32968         shadow:true,
32969         minWidth:500,
32970         minHeight:350,
32971         autoTabs:true,
32972         proxyDrag:true,
32973         // layout config merges with the dialog config
32974         center:{
32975             tabPosition: "top",
32976             alwaysShowTabs: true
32977         }
32978     });
32979     dialog.addKeyListener(27, dialog.hide, dialog);
32980     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
32981     dialog.addButton("Build It!", this.getDownload, this);
32982
32983     // we can even add nested layouts
32984     var innerLayout = new Roo.BorderLayout("dl-inner", {
32985         east: {
32986             initialSize: 200,
32987             autoScroll:true,
32988             split:true
32989         },
32990         center: {
32991             autoScroll:true
32992         }
32993     });
32994     innerLayout.beginUpdate();
32995     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
32996     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
32997     innerLayout.endUpdate(true);
32998
32999     var layout = dialog.getLayout();
33000     layout.beginUpdate();
33001     layout.add("center", new Roo.ContentPanel("standard-panel",
33002                         {title: "Download the Source", fitToFrame:true}));
33003     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33004                {title: "Build your own roo.js"}));
33005     layout.getRegion("center").showPanel(sp);
33006     layout.endUpdate();
33007 }
33008 </code></pre>
33009     * @constructor
33010     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33011     * @param {Object} config configuration options
33012   */
33013 Roo.LayoutDialog = function(el, cfg){
33014     
33015     var config=  cfg;
33016     if (typeof(cfg) == 'undefined') {
33017         config = Roo.apply({}, el);
33018         // not sure why we use documentElement here.. - it should always be body.
33019         // IE7 borks horribly if we use documentElement.
33020         // webkit also does not like documentElement - it creates a body element...
33021         el = Roo.get( document.body || document.documentElement ).createChild();
33022         //config.autoCreate = true;
33023     }
33024     
33025     
33026     config.autoTabs = false;
33027     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33028     this.body.setStyle({overflow:"hidden", position:"relative"});
33029     this.layout = new Roo.BorderLayout(this.body.dom, config);
33030     this.layout.monitorWindowResize = false;
33031     this.el.addClass("x-dlg-auto-layout");
33032     // fix case when center region overwrites center function
33033     this.center = Roo.BasicDialog.prototype.center;
33034     this.on("show", this.layout.layout, this.layout, true);
33035     if (config.items) {
33036         var xitems = config.items;
33037         delete config.items;
33038         Roo.each(xitems, this.addxtype, this);
33039     }
33040     
33041     
33042 };
33043 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33044     /**
33045      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33046      * @deprecated
33047      */
33048     endUpdate : function(){
33049         this.layout.endUpdate();
33050     },
33051
33052     /**
33053      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33054      *  @deprecated
33055      */
33056     beginUpdate : function(){
33057         this.layout.beginUpdate();
33058     },
33059
33060     /**
33061      * Get the BorderLayout for this dialog
33062      * @return {Roo.BorderLayout}
33063      */
33064     getLayout : function(){
33065         return this.layout;
33066     },
33067
33068     showEl : function(){
33069         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33070         if(Roo.isIE7){
33071             this.layout.layout();
33072         }
33073     },
33074
33075     // private
33076     // Use the syncHeightBeforeShow config option to control this automatically
33077     syncBodyHeight : function(){
33078         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33079         if(this.layout){this.layout.layout();}
33080     },
33081     
33082       /**
33083      * Add an xtype element (actually adds to the layout.)
33084      * @return {Object} xdata xtype object data.
33085      */
33086     
33087     addxtype : function(c) {
33088         return this.layout.addxtype(c);
33089     }
33090 });/*
33091  * Based on:
33092  * Ext JS Library 1.1.1
33093  * Copyright(c) 2006-2007, Ext JS, LLC.
33094  *
33095  * Originally Released Under LGPL - original licence link has changed is not relivant.
33096  *
33097  * Fork - LGPL
33098  * <script type="text/javascript">
33099  */
33100  
33101 /**
33102  * @class Roo.MessageBox
33103  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33104  * Example usage:
33105  *<pre><code>
33106 // Basic alert:
33107 Roo.Msg.alert('Status', 'Changes saved successfully.');
33108
33109 // Prompt for user data:
33110 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33111     if (btn == 'ok'){
33112         // process text value...
33113     }
33114 });
33115
33116 // Show a dialog using config options:
33117 Roo.Msg.show({
33118    title:'Save Changes?',
33119    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33120    buttons: Roo.Msg.YESNOCANCEL,
33121    fn: processResult,
33122    animEl: 'elId'
33123 });
33124 </code></pre>
33125  * @singleton
33126  */
33127 Roo.MessageBox = function(){
33128     var dlg, opt, mask, waitTimer;
33129     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33130     var buttons, activeTextEl, bwidth;
33131
33132     // private
33133     var handleButton = function(button){
33134         dlg.hide();
33135         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33136     };
33137
33138     // private
33139     var handleHide = function(){
33140         if(opt && opt.cls){
33141             dlg.el.removeClass(opt.cls);
33142         }
33143         if(waitTimer){
33144             Roo.TaskMgr.stop(waitTimer);
33145             waitTimer = null;
33146         }
33147     };
33148
33149     // private
33150     var updateButtons = function(b){
33151         var width = 0;
33152         if(!b){
33153             buttons["ok"].hide();
33154             buttons["cancel"].hide();
33155             buttons["yes"].hide();
33156             buttons["no"].hide();
33157             dlg.footer.dom.style.display = 'none';
33158             return width;
33159         }
33160         dlg.footer.dom.style.display = '';
33161         for(var k in buttons){
33162             if(typeof buttons[k] != "function"){
33163                 if(b[k]){
33164                     buttons[k].show();
33165                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33166                     width += buttons[k].el.getWidth()+15;
33167                 }else{
33168                     buttons[k].hide();
33169                 }
33170             }
33171         }
33172         return width;
33173     };
33174
33175     // private
33176     var handleEsc = function(d, k, e){
33177         if(opt && opt.closable !== false){
33178             dlg.hide();
33179         }
33180         if(e){
33181             e.stopEvent();
33182         }
33183     };
33184
33185     return {
33186         /**
33187          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33188          * @return {Roo.BasicDialog} The BasicDialog element
33189          */
33190         getDialog : function(){
33191            if(!dlg){
33192                 dlg = new Roo.BasicDialog("x-msg-box", {
33193                     autoCreate : true,
33194                     shadow: true,
33195                     draggable: true,
33196                     resizable:false,
33197                     constraintoviewport:false,
33198                     fixedcenter:true,
33199                     collapsible : false,
33200                     shim:true,
33201                     modal: true,
33202                     width:400, height:100,
33203                     buttonAlign:"center",
33204                     closeClick : function(){
33205                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33206                             handleButton("no");
33207                         }else{
33208                             handleButton("cancel");
33209                         }
33210                     }
33211                 });
33212                 dlg.on("hide", handleHide);
33213                 mask = dlg.mask;
33214                 dlg.addKeyListener(27, handleEsc);
33215                 buttons = {};
33216                 var bt = this.buttonText;
33217                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33218                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33219                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33220                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33221                 bodyEl = dlg.body.createChild({
33222
33223                     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>'
33224                 });
33225                 msgEl = bodyEl.dom.firstChild;
33226                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33227                 textboxEl.enableDisplayMode();
33228                 textboxEl.addKeyListener([10,13], function(){
33229                     if(dlg.isVisible() && opt && opt.buttons){
33230                         if(opt.buttons.ok){
33231                             handleButton("ok");
33232                         }else if(opt.buttons.yes){
33233                             handleButton("yes");
33234                         }
33235                     }
33236                 });
33237                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33238                 textareaEl.enableDisplayMode();
33239                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33240                 progressEl.enableDisplayMode();
33241                 var pf = progressEl.dom.firstChild;
33242                 if (pf) {
33243                     pp = Roo.get(pf.firstChild);
33244                     pp.setHeight(pf.offsetHeight);
33245                 }
33246                 
33247             }
33248             return dlg;
33249         },
33250
33251         /**
33252          * Updates the message box body text
33253          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33254          * the XHTML-compliant non-breaking space character '&amp;#160;')
33255          * @return {Roo.MessageBox} This message box
33256          */
33257         updateText : function(text){
33258             if(!dlg.isVisible() && !opt.width){
33259                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33260             }
33261             msgEl.innerHTML = text || '&#160;';
33262       
33263             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33264             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33265             var w = Math.max(
33266                     Math.min(opt.width || cw , this.maxWidth), 
33267                     Math.max(opt.minWidth || this.minWidth, bwidth)
33268             );
33269             if(opt.prompt){
33270                 activeTextEl.setWidth(w);
33271             }
33272             if(dlg.isVisible()){
33273                 dlg.fixedcenter = false;
33274             }
33275             // to big, make it scroll. = But as usual stupid IE does not support
33276             // !important..
33277             
33278             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33279                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33280                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33281             } else {
33282                 bodyEl.dom.style.height = '';
33283                 bodyEl.dom.style.overflowY = '';
33284             }
33285             if (cw > w) {
33286                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33287             } else {
33288                 bodyEl.dom.style.overflowX = '';
33289             }
33290             
33291             dlg.setContentSize(w, bodyEl.getHeight());
33292             if(dlg.isVisible()){
33293                 dlg.fixedcenter = true;
33294             }
33295             return this;
33296         },
33297
33298         /**
33299          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33300          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33301          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33302          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33303          * @return {Roo.MessageBox} This message box
33304          */
33305         updateProgress : function(value, text){
33306             if(text){
33307                 this.updateText(text);
33308             }
33309             if (pp) { // weird bug on my firefox - for some reason this is not defined
33310                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33311             }
33312             return this;
33313         },        
33314
33315         /**
33316          * Returns true if the message box is currently displayed
33317          * @return {Boolean} True if the message box is visible, else false
33318          */
33319         isVisible : function(){
33320             return dlg && dlg.isVisible();  
33321         },
33322
33323         /**
33324          * Hides the message box if it is displayed
33325          */
33326         hide : function(){
33327             if(this.isVisible()){
33328                 dlg.hide();
33329             }  
33330         },
33331
33332         /**
33333          * Displays a new message box, or reinitializes an existing message box, based on the config options
33334          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33335          * The following config object properties are supported:
33336          * <pre>
33337 Property    Type             Description
33338 ----------  ---------------  ------------------------------------------------------------------------------------
33339 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33340                                    closes (defaults to undefined)
33341 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33342                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33343 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33344                                    progress and wait dialogs will ignore this property and always hide the
33345                                    close button as they can only be closed programmatically.
33346 cls               String           A custom CSS class to apply to the message box element
33347 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33348                                    displayed (defaults to 75)
33349 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33350                                    function will be btn (the name of the button that was clicked, if applicable,
33351                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33352                                    Progress and wait dialogs will ignore this option since they do not respond to
33353                                    user actions and can only be closed programmatically, so any required function
33354                                    should be called by the same code after it closes the dialog.
33355 icon              String           A CSS class that provides a background image to be used as an icon for
33356                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33357 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33358 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33359 modal             Boolean          False to allow user interaction with the page while the message box is
33360                                    displayed (defaults to true)
33361 msg               String           A string that will replace the existing message box body text (defaults
33362                                    to the XHTML-compliant non-breaking space character '&#160;')
33363 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33364 progress          Boolean          True to display a progress bar (defaults to false)
33365 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33366 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33367 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33368 title             String           The title text
33369 value             String           The string value to set into the active textbox element if displayed
33370 wait              Boolean          True to display a progress bar (defaults to false)
33371 width             Number           The width of the dialog in pixels
33372 </pre>
33373          *
33374          * Example usage:
33375          * <pre><code>
33376 Roo.Msg.show({
33377    title: 'Address',
33378    msg: 'Please enter your address:',
33379    width: 300,
33380    buttons: Roo.MessageBox.OKCANCEL,
33381    multiline: true,
33382    fn: saveAddress,
33383    animEl: 'addAddressBtn'
33384 });
33385 </code></pre>
33386          * @param {Object} config Configuration options
33387          * @return {Roo.MessageBox} This message box
33388          */
33389         show : function(options)
33390         {
33391             
33392             // this causes nightmares if you show one dialog after another
33393             // especially on callbacks..
33394              
33395             if(this.isVisible()){
33396                 
33397                 this.hide();
33398                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33399                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33400                 Roo.log("New Dialog Message:" +  options.msg )
33401                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33402                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33403                 
33404             }
33405             var d = this.getDialog();
33406             opt = options;
33407             d.setTitle(opt.title || "&#160;");
33408             d.close.setDisplayed(opt.closable !== false);
33409             activeTextEl = textboxEl;
33410             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33411             if(opt.prompt){
33412                 if(opt.multiline){
33413                     textboxEl.hide();
33414                     textareaEl.show();
33415                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33416                         opt.multiline : this.defaultTextHeight);
33417                     activeTextEl = textareaEl;
33418                 }else{
33419                     textboxEl.show();
33420                     textareaEl.hide();
33421                 }
33422             }else{
33423                 textboxEl.hide();
33424                 textareaEl.hide();
33425             }
33426             progressEl.setDisplayed(opt.progress === true);
33427             this.updateProgress(0);
33428             activeTextEl.dom.value = opt.value || "";
33429             if(opt.prompt){
33430                 dlg.setDefaultButton(activeTextEl);
33431             }else{
33432                 var bs = opt.buttons;
33433                 var db = null;
33434                 if(bs && bs.ok){
33435                     db = buttons["ok"];
33436                 }else if(bs && bs.yes){
33437                     db = buttons["yes"];
33438                 }
33439                 dlg.setDefaultButton(db);
33440             }
33441             bwidth = updateButtons(opt.buttons);
33442             this.updateText(opt.msg);
33443             if(opt.cls){
33444                 d.el.addClass(opt.cls);
33445             }
33446             d.proxyDrag = opt.proxyDrag === true;
33447             d.modal = opt.modal !== false;
33448             d.mask = opt.modal !== false ? mask : false;
33449             if(!d.isVisible()){
33450                 // force it to the end of the z-index stack so it gets a cursor in FF
33451                 document.body.appendChild(dlg.el.dom);
33452                 d.animateTarget = null;
33453                 d.show(options.animEl);
33454             }
33455             return this;
33456         },
33457
33458         /**
33459          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33460          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33461          * and closing the message box when the process is complete.
33462          * @param {String} title The title bar text
33463          * @param {String} msg The message box body text
33464          * @return {Roo.MessageBox} This message box
33465          */
33466         progress : function(title, msg){
33467             this.show({
33468                 title : title,
33469                 msg : msg,
33470                 buttons: false,
33471                 progress:true,
33472                 closable:false,
33473                 minWidth: this.minProgressWidth,
33474                 modal : true
33475             });
33476             return this;
33477         },
33478
33479         /**
33480          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33481          * If a callback function is passed it will be called after the user clicks the button, and the
33482          * id of the button that was clicked will be passed as the only parameter to the callback
33483          * (could also be the top-right close button).
33484          * @param {String} title The title bar text
33485          * @param {String} msg The message box body text
33486          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33487          * @param {Object} scope (optional) The scope of the callback function
33488          * @return {Roo.MessageBox} This message box
33489          */
33490         alert : function(title, msg, fn, scope){
33491             this.show({
33492                 title : title,
33493                 msg : msg,
33494                 buttons: this.OK,
33495                 fn: fn,
33496                 scope : scope,
33497                 modal : true
33498             });
33499             return this;
33500         },
33501
33502         /**
33503          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33504          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33505          * You are responsible for closing the message box when the process is complete.
33506          * @param {String} msg The message box body text
33507          * @param {String} title (optional) The title bar text
33508          * @return {Roo.MessageBox} This message box
33509          */
33510         wait : function(msg, title){
33511             this.show({
33512                 title : title,
33513                 msg : msg,
33514                 buttons: false,
33515                 closable:false,
33516                 progress:true,
33517                 modal:true,
33518                 width:300,
33519                 wait:true
33520             });
33521             waitTimer = Roo.TaskMgr.start({
33522                 run: function(i){
33523                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33524                 },
33525                 interval: 1000
33526             });
33527             return this;
33528         },
33529
33530         /**
33531          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33532          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33533          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33534          * @param {String} title The title bar text
33535          * @param {String} msg The message box body text
33536          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33537          * @param {Object} scope (optional) The scope of the callback function
33538          * @return {Roo.MessageBox} This message box
33539          */
33540         confirm : function(title, msg, fn, scope){
33541             this.show({
33542                 title : title,
33543                 msg : msg,
33544                 buttons: this.YESNO,
33545                 fn: fn,
33546                 scope : scope,
33547                 modal : true
33548             });
33549             return this;
33550         },
33551
33552         /**
33553          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33554          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33555          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33556          * (could also be the top-right close button) and the text that was entered will be passed as the two
33557          * parameters to the callback.
33558          * @param {String} title The title bar text
33559          * @param {String} msg The message box body text
33560          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33561          * @param {Object} scope (optional) The scope of the callback function
33562          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33563          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33564          * @return {Roo.MessageBox} This message box
33565          */
33566         prompt : function(title, msg, fn, scope, multiline){
33567             this.show({
33568                 title : title,
33569                 msg : msg,
33570                 buttons: this.OKCANCEL,
33571                 fn: fn,
33572                 minWidth:250,
33573                 scope : scope,
33574                 prompt:true,
33575                 multiline: multiline,
33576                 modal : true
33577             });
33578             return this;
33579         },
33580
33581         /**
33582          * Button config that displays a single OK button
33583          * @type Object
33584          */
33585         OK : {ok:true},
33586         /**
33587          * Button config that displays Yes and No buttons
33588          * @type Object
33589          */
33590         YESNO : {yes:true, no:true},
33591         /**
33592          * Button config that displays OK and Cancel buttons
33593          * @type Object
33594          */
33595         OKCANCEL : {ok:true, cancel:true},
33596         /**
33597          * Button config that displays Yes, No and Cancel buttons
33598          * @type Object
33599          */
33600         YESNOCANCEL : {yes:true, no:true, cancel:true},
33601
33602         /**
33603          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33604          * @type Number
33605          */
33606         defaultTextHeight : 75,
33607         /**
33608          * The maximum width in pixels of the message box (defaults to 600)
33609          * @type Number
33610          */
33611         maxWidth : 600,
33612         /**
33613          * The minimum width in pixels of the message box (defaults to 100)
33614          * @type Number
33615          */
33616         minWidth : 100,
33617         /**
33618          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33619          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33620          * @type Number
33621          */
33622         minProgressWidth : 250,
33623         /**
33624          * An object containing the default button text strings that can be overriden for localized language support.
33625          * Supported properties are: ok, cancel, yes and no.
33626          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33627          * @type Object
33628          */
33629         buttonText : {
33630             ok : "OK",
33631             cancel : "Cancel",
33632             yes : "Yes",
33633             no : "No"
33634         }
33635     };
33636 }();
33637
33638 /**
33639  * Shorthand for {@link Roo.MessageBox}
33640  */
33641 Roo.Msg = Roo.MessageBox;/*
33642  * Based on:
33643  * Ext JS Library 1.1.1
33644  * Copyright(c) 2006-2007, Ext JS, LLC.
33645  *
33646  * Originally Released Under LGPL - original licence link has changed is not relivant.
33647  *
33648  * Fork - LGPL
33649  * <script type="text/javascript">
33650  */
33651 /**
33652  * @class Roo.QuickTips
33653  * Provides attractive and customizable tooltips for any element.
33654  * @singleton
33655  */
33656 Roo.QuickTips = function(){
33657     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33658     var ce, bd, xy, dd;
33659     var visible = false, disabled = true, inited = false;
33660     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33661     
33662     var onOver = function(e){
33663         if(disabled){
33664             return;
33665         }
33666         var t = e.getTarget();
33667         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33668             return;
33669         }
33670         if(ce && t == ce.el){
33671             clearTimeout(hideProc);
33672             return;
33673         }
33674         if(t && tagEls[t.id]){
33675             tagEls[t.id].el = t;
33676             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33677             return;
33678         }
33679         var ttp, et = Roo.fly(t);
33680         var ns = cfg.namespace;
33681         if(tm.interceptTitles && t.title){
33682             ttp = t.title;
33683             t.qtip = ttp;
33684             t.removeAttribute("title");
33685             e.preventDefault();
33686         }else{
33687             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33688         }
33689         if(ttp){
33690             showProc = show.defer(tm.showDelay, tm, [{
33691                 el: t, 
33692                 text: ttp.replace(/\\n/g,'<br/>'),
33693                 width: et.getAttributeNS(ns, cfg.width),
33694                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33695                 title: et.getAttributeNS(ns, cfg.title),
33696                     cls: et.getAttributeNS(ns, cfg.cls)
33697             }]);
33698         }
33699     };
33700     
33701     var onOut = function(e){
33702         clearTimeout(showProc);
33703         var t = e.getTarget();
33704         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33705             hideProc = setTimeout(hide, tm.hideDelay);
33706         }
33707     };
33708     
33709     var onMove = function(e){
33710         if(disabled){
33711             return;
33712         }
33713         xy = e.getXY();
33714         xy[1] += 18;
33715         if(tm.trackMouse && ce){
33716             el.setXY(xy);
33717         }
33718     };
33719     
33720     var onDown = function(e){
33721         clearTimeout(showProc);
33722         clearTimeout(hideProc);
33723         if(!e.within(el)){
33724             if(tm.hideOnClick){
33725                 hide();
33726                 tm.disable();
33727                 tm.enable.defer(100, tm);
33728             }
33729         }
33730     };
33731     
33732     var getPad = function(){
33733         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33734     };
33735
33736     var show = function(o){
33737         if(disabled){
33738             return;
33739         }
33740         clearTimeout(dismissProc);
33741         ce = o;
33742         if(removeCls){ // in case manually hidden
33743             el.removeClass(removeCls);
33744             removeCls = null;
33745         }
33746         if(ce.cls){
33747             el.addClass(ce.cls);
33748             removeCls = ce.cls;
33749         }
33750         if(ce.title){
33751             tipTitle.update(ce.title);
33752             tipTitle.show();
33753         }else{
33754             tipTitle.update('');
33755             tipTitle.hide();
33756         }
33757         el.dom.style.width  = tm.maxWidth+'px';
33758         //tipBody.dom.style.width = '';
33759         tipBodyText.update(o.text);
33760         var p = getPad(), w = ce.width;
33761         if(!w){
33762             var td = tipBodyText.dom;
33763             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33764             if(aw > tm.maxWidth){
33765                 w = tm.maxWidth;
33766             }else if(aw < tm.minWidth){
33767                 w = tm.minWidth;
33768             }else{
33769                 w = aw;
33770             }
33771         }
33772         //tipBody.setWidth(w);
33773         el.setWidth(parseInt(w, 10) + p);
33774         if(ce.autoHide === false){
33775             close.setDisplayed(true);
33776             if(dd){
33777                 dd.unlock();
33778             }
33779         }else{
33780             close.setDisplayed(false);
33781             if(dd){
33782                 dd.lock();
33783             }
33784         }
33785         if(xy){
33786             el.avoidY = xy[1]-18;
33787             el.setXY(xy);
33788         }
33789         if(tm.animate){
33790             el.setOpacity(.1);
33791             el.setStyle("visibility", "visible");
33792             el.fadeIn({callback: afterShow});
33793         }else{
33794             afterShow();
33795         }
33796     };
33797     
33798     var afterShow = function(){
33799         if(ce){
33800             el.show();
33801             esc.enable();
33802             if(tm.autoDismiss && ce.autoHide !== false){
33803                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33804             }
33805         }
33806     };
33807     
33808     var hide = function(noanim){
33809         clearTimeout(dismissProc);
33810         clearTimeout(hideProc);
33811         ce = null;
33812         if(el.isVisible()){
33813             esc.disable();
33814             if(noanim !== true && tm.animate){
33815                 el.fadeOut({callback: afterHide});
33816             }else{
33817                 afterHide();
33818             } 
33819         }
33820     };
33821     
33822     var afterHide = function(){
33823         el.hide();
33824         if(removeCls){
33825             el.removeClass(removeCls);
33826             removeCls = null;
33827         }
33828     };
33829     
33830     return {
33831         /**
33832         * @cfg {Number} minWidth
33833         * The minimum width of the quick tip (defaults to 40)
33834         */
33835        minWidth : 40,
33836         /**
33837         * @cfg {Number} maxWidth
33838         * The maximum width of the quick tip (defaults to 300)
33839         */
33840        maxWidth : 300,
33841         /**
33842         * @cfg {Boolean} interceptTitles
33843         * True to automatically use the element's DOM title value if available (defaults to false)
33844         */
33845        interceptTitles : false,
33846         /**
33847         * @cfg {Boolean} trackMouse
33848         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33849         */
33850        trackMouse : false,
33851         /**
33852         * @cfg {Boolean} hideOnClick
33853         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33854         */
33855        hideOnClick : true,
33856         /**
33857         * @cfg {Number} showDelay
33858         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33859         */
33860        showDelay : 500,
33861         /**
33862         * @cfg {Number} hideDelay
33863         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33864         */
33865        hideDelay : 200,
33866         /**
33867         * @cfg {Boolean} autoHide
33868         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33869         * Used in conjunction with hideDelay.
33870         */
33871        autoHide : true,
33872         /**
33873         * @cfg {Boolean}
33874         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33875         * (defaults to true).  Used in conjunction with autoDismissDelay.
33876         */
33877        autoDismiss : true,
33878         /**
33879         * @cfg {Number}
33880         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33881         */
33882        autoDismissDelay : 5000,
33883        /**
33884         * @cfg {Boolean} animate
33885         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33886         */
33887        animate : false,
33888
33889        /**
33890         * @cfg {String} title
33891         * Title text to display (defaults to '').  This can be any valid HTML markup.
33892         */
33893         title: '',
33894        /**
33895         * @cfg {String} text
33896         * Body text to display (defaults to '').  This can be any valid HTML markup.
33897         */
33898         text : '',
33899        /**
33900         * @cfg {String} cls
33901         * A CSS class to apply to the base quick tip element (defaults to '').
33902         */
33903         cls : '',
33904        /**
33905         * @cfg {Number} width
33906         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33907         * minWidth or maxWidth.
33908         */
33909         width : null,
33910
33911     /**
33912      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33913      * or display QuickTips in a page.
33914      */
33915        init : function(){
33916           tm = Roo.QuickTips;
33917           cfg = tm.tagConfig;
33918           if(!inited){
33919               if(!Roo.isReady){ // allow calling of init() before onReady
33920                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33921                   return;
33922               }
33923               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33924               el.fxDefaults = {stopFx: true};
33925               // maximum custom styling
33926               //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>');
33927               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>');              
33928               tipTitle = el.child('h3');
33929               tipTitle.enableDisplayMode("block");
33930               tipBody = el.child('div.x-tip-bd');
33931               tipBodyText = el.child('div.x-tip-bd-inner');
33932               //bdLeft = el.child('div.x-tip-bd-left');
33933               //bdRight = el.child('div.x-tip-bd-right');
33934               close = el.child('div.x-tip-close');
33935               close.enableDisplayMode("block");
33936               close.on("click", hide);
33937               var d = Roo.get(document);
33938               d.on("mousedown", onDown);
33939               d.on("mouseover", onOver);
33940               d.on("mouseout", onOut);
33941               d.on("mousemove", onMove);
33942               esc = d.addKeyListener(27, hide);
33943               esc.disable();
33944               if(Roo.dd.DD){
33945                   dd = el.initDD("default", null, {
33946                       onDrag : function(){
33947                           el.sync();  
33948                       }
33949                   });
33950                   dd.setHandleElId(tipTitle.id);
33951                   dd.lock();
33952               }
33953               inited = true;
33954           }
33955           this.enable(); 
33956        },
33957
33958     /**
33959      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33960      * are supported:
33961      * <pre>
33962 Property    Type                   Description
33963 ----------  ---------------------  ------------------------------------------------------------------------
33964 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
33965      * </ul>
33966      * @param {Object} config The config object
33967      */
33968        register : function(config){
33969            var cs = config instanceof Array ? config : arguments;
33970            for(var i = 0, len = cs.length; i < len; i++) {
33971                var c = cs[i];
33972                var target = c.target;
33973                if(target){
33974                    if(target instanceof Array){
33975                        for(var j = 0, jlen = target.length; j < jlen; j++){
33976                            tagEls[target[j]] = c;
33977                        }
33978                    }else{
33979                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
33980                    }
33981                }
33982            }
33983        },
33984
33985     /**
33986      * Removes this quick tip from its element and destroys it.
33987      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
33988      */
33989        unregister : function(el){
33990            delete tagEls[Roo.id(el)];
33991        },
33992
33993     /**
33994      * Enable this quick tip.
33995      */
33996        enable : function(){
33997            if(inited && disabled){
33998                locks.pop();
33999                if(locks.length < 1){
34000                    disabled = false;
34001                }
34002            }
34003        },
34004
34005     /**
34006      * Disable this quick tip.
34007      */
34008        disable : function(){
34009           disabled = true;
34010           clearTimeout(showProc);
34011           clearTimeout(hideProc);
34012           clearTimeout(dismissProc);
34013           if(ce){
34014               hide(true);
34015           }
34016           locks.push(1);
34017        },
34018
34019     /**
34020      * Returns true if the quick tip is enabled, else false.
34021      */
34022        isEnabled : function(){
34023             return !disabled;
34024        },
34025
34026         // private
34027        tagConfig : {
34028            namespace : "roo", // was ext?? this may break..
34029            alt_namespace : "ext",
34030            attribute : "qtip",
34031            width : "width",
34032            target : "target",
34033            title : "qtitle",
34034            hide : "hide",
34035            cls : "qclass"
34036        }
34037    };
34038 }();
34039
34040 // backwards compat
34041 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34042  * Based on:
34043  * Ext JS Library 1.1.1
34044  * Copyright(c) 2006-2007, Ext JS, LLC.
34045  *
34046  * Originally Released Under LGPL - original licence link has changed is not relivant.
34047  *
34048  * Fork - LGPL
34049  * <script type="text/javascript">
34050  */
34051  
34052
34053 /**
34054  * @class Roo.tree.TreePanel
34055  * @extends Roo.data.Tree
34056
34057  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34058  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34059  * @cfg {Boolean} enableDD true to enable drag and drop
34060  * @cfg {Boolean} enableDrag true to enable just drag
34061  * @cfg {Boolean} enableDrop true to enable just drop
34062  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34063  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34064  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34065  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34066  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34067  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34068  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34069  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34070  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34071  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34072  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34073  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34074  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34075  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34076  * @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>
34077  * @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>
34078  * 
34079  * @constructor
34080  * @param {String/HTMLElement/Element} el The container element
34081  * @param {Object} config
34082  */
34083 Roo.tree.TreePanel = function(el, config){
34084     var root = false;
34085     var loader = false;
34086     if (config.root) {
34087         root = config.root;
34088         delete config.root;
34089     }
34090     if (config.loader) {
34091         loader = config.loader;
34092         delete config.loader;
34093     }
34094     
34095     Roo.apply(this, config);
34096     Roo.tree.TreePanel.superclass.constructor.call(this);
34097     this.el = Roo.get(el);
34098     this.el.addClass('x-tree');
34099     //console.log(root);
34100     if (root) {
34101         this.setRootNode( Roo.factory(root, Roo.tree));
34102     }
34103     if (loader) {
34104         this.loader = Roo.factory(loader, Roo.tree);
34105     }
34106    /**
34107     * Read-only. The id of the container element becomes this TreePanel's id.
34108     */
34109     this.id = this.el.id;
34110     this.addEvents({
34111         /**
34112         * @event beforeload
34113         * Fires before a node is loaded, return false to cancel
34114         * @param {Node} node The node being loaded
34115         */
34116         "beforeload" : true,
34117         /**
34118         * @event load
34119         * Fires when a node is loaded
34120         * @param {Node} node The node that was loaded
34121         */
34122         "load" : true,
34123         /**
34124         * @event textchange
34125         * Fires when the text for a node is changed
34126         * @param {Node} node The node
34127         * @param {String} text The new text
34128         * @param {String} oldText The old text
34129         */
34130         "textchange" : true,
34131         /**
34132         * @event beforeexpand
34133         * Fires before a node is expanded, return false to cancel.
34134         * @param {Node} node The node
34135         * @param {Boolean} deep
34136         * @param {Boolean} anim
34137         */
34138         "beforeexpand" : true,
34139         /**
34140         * @event beforecollapse
34141         * Fires before a node is collapsed, return false to cancel.
34142         * @param {Node} node The node
34143         * @param {Boolean} deep
34144         * @param {Boolean} anim
34145         */
34146         "beforecollapse" : true,
34147         /**
34148         * @event expand
34149         * Fires when a node is expanded
34150         * @param {Node} node The node
34151         */
34152         "expand" : true,
34153         /**
34154         * @event disabledchange
34155         * Fires when the disabled status of a node changes
34156         * @param {Node} node The node
34157         * @param {Boolean} disabled
34158         */
34159         "disabledchange" : true,
34160         /**
34161         * @event collapse
34162         * Fires when a node is collapsed
34163         * @param {Node} node The node
34164         */
34165         "collapse" : true,
34166         /**
34167         * @event beforeclick
34168         * Fires before click processing on a node. Return false to cancel the default action.
34169         * @param {Node} node The node
34170         * @param {Roo.EventObject} e The event object
34171         */
34172         "beforeclick":true,
34173         /**
34174         * @event checkchange
34175         * Fires when a node with a checkbox's checked property changes
34176         * @param {Node} this This node
34177         * @param {Boolean} checked
34178         */
34179         "checkchange":true,
34180         /**
34181         * @event click
34182         * Fires when a node is clicked
34183         * @param {Node} node The node
34184         * @param {Roo.EventObject} e The event object
34185         */
34186         "click":true,
34187         /**
34188         * @event dblclick
34189         * Fires when a node is double clicked
34190         * @param {Node} node The node
34191         * @param {Roo.EventObject} e The event object
34192         */
34193         "dblclick":true,
34194         /**
34195         * @event contextmenu
34196         * Fires when a node is right clicked
34197         * @param {Node} node The node
34198         * @param {Roo.EventObject} e The event object
34199         */
34200         "contextmenu":true,
34201         /**
34202         * @event beforechildrenrendered
34203         * Fires right before the child nodes for a node are rendered
34204         * @param {Node} node The node
34205         */
34206         "beforechildrenrendered":true,
34207         /**
34208         * @event startdrag
34209         * Fires when a node starts being dragged
34210         * @param {Roo.tree.TreePanel} this
34211         * @param {Roo.tree.TreeNode} node
34212         * @param {event} e The raw browser event
34213         */ 
34214        "startdrag" : true,
34215        /**
34216         * @event enddrag
34217         * Fires when a drag operation is complete
34218         * @param {Roo.tree.TreePanel} this
34219         * @param {Roo.tree.TreeNode} node
34220         * @param {event} e The raw browser event
34221         */
34222        "enddrag" : true,
34223        /**
34224         * @event dragdrop
34225         * Fires when a dragged node is dropped on a valid DD target
34226         * @param {Roo.tree.TreePanel} this
34227         * @param {Roo.tree.TreeNode} node
34228         * @param {DD} dd The dd it was dropped on
34229         * @param {event} e The raw browser event
34230         */
34231        "dragdrop" : true,
34232        /**
34233         * @event beforenodedrop
34234         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34235         * passed to handlers has the following properties:<br />
34236         * <ul style="padding:5px;padding-left:16px;">
34237         * <li>tree - The TreePanel</li>
34238         * <li>target - The node being targeted for the drop</li>
34239         * <li>data - The drag data from the drag source</li>
34240         * <li>point - The point of the drop - append, above or below</li>
34241         * <li>source - The drag source</li>
34242         * <li>rawEvent - Raw mouse event</li>
34243         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34244         * to be inserted by setting them on this object.</li>
34245         * <li>cancel - Set this to true to cancel the drop.</li>
34246         * </ul>
34247         * @param {Object} dropEvent
34248         */
34249        "beforenodedrop" : true,
34250        /**
34251         * @event nodedrop
34252         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34253         * passed to handlers has the following properties:<br />
34254         * <ul style="padding:5px;padding-left:16px;">
34255         * <li>tree - The TreePanel</li>
34256         * <li>target - The node being targeted for the drop</li>
34257         * <li>data - The drag data from the drag source</li>
34258         * <li>point - The point of the drop - append, above or below</li>
34259         * <li>source - The drag source</li>
34260         * <li>rawEvent - Raw mouse event</li>
34261         * <li>dropNode - Dropped node(s).</li>
34262         * </ul>
34263         * @param {Object} dropEvent
34264         */
34265        "nodedrop" : true,
34266         /**
34267         * @event nodedragover
34268         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34269         * passed to handlers has the following properties:<br />
34270         * <ul style="padding:5px;padding-left:16px;">
34271         * <li>tree - The TreePanel</li>
34272         * <li>target - The node being targeted for the drop</li>
34273         * <li>data - The drag data from the drag source</li>
34274         * <li>point - The point of the drop - append, above or below</li>
34275         * <li>source - The drag source</li>
34276         * <li>rawEvent - Raw mouse event</li>
34277         * <li>dropNode - Drop node(s) provided by the source.</li>
34278         * <li>cancel - Set this to true to signal drop not allowed.</li>
34279         * </ul>
34280         * @param {Object} dragOverEvent
34281         */
34282        "nodedragover" : true,
34283        /**
34284         * @event appendnode
34285         * Fires when append node to the tree
34286         * @param {Roo.tree.TreePanel} this
34287         * @param {Roo.tree.TreeNode} node
34288         * @param {Number} index The index of the newly appended node
34289         */
34290        "appendnode" : true
34291         
34292     });
34293     if(this.singleExpand){
34294        this.on("beforeexpand", this.restrictExpand, this);
34295     }
34296     if (this.editor) {
34297         this.editor.tree = this;
34298         this.editor = Roo.factory(this.editor, Roo.tree);
34299     }
34300     
34301     if (this.selModel) {
34302         this.selModel = Roo.factory(this.selModel, Roo.tree);
34303     }
34304    
34305 };
34306 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34307     rootVisible : true,
34308     animate: Roo.enableFx,
34309     lines : true,
34310     enableDD : false,
34311     hlDrop : Roo.enableFx,
34312   
34313     renderer: false,
34314     
34315     rendererTip: false,
34316     // private
34317     restrictExpand : function(node){
34318         var p = node.parentNode;
34319         if(p){
34320             if(p.expandedChild && p.expandedChild.parentNode == p){
34321                 p.expandedChild.collapse();
34322             }
34323             p.expandedChild = node;
34324         }
34325     },
34326
34327     // private override
34328     setRootNode : function(node){
34329         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34330         if(!this.rootVisible){
34331             node.ui = new Roo.tree.RootTreeNodeUI(node);
34332         }
34333         return node;
34334     },
34335
34336     /**
34337      * Returns the container element for this TreePanel
34338      */
34339     getEl : function(){
34340         return this.el;
34341     },
34342
34343     /**
34344      * Returns the default TreeLoader for this TreePanel
34345      */
34346     getLoader : function(){
34347         return this.loader;
34348     },
34349
34350     /**
34351      * Expand all nodes
34352      */
34353     expandAll : function(){
34354         this.root.expand(true);
34355     },
34356
34357     /**
34358      * Collapse all nodes
34359      */
34360     collapseAll : function(){
34361         this.root.collapse(true);
34362     },
34363
34364     /**
34365      * Returns the selection model used by this TreePanel
34366      */
34367     getSelectionModel : function(){
34368         if(!this.selModel){
34369             this.selModel = new Roo.tree.DefaultSelectionModel();
34370         }
34371         return this.selModel;
34372     },
34373
34374     /**
34375      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34376      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34377      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34378      * @return {Array}
34379      */
34380     getChecked : function(a, startNode){
34381         startNode = startNode || this.root;
34382         var r = [];
34383         var f = function(){
34384             if(this.attributes.checked){
34385                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34386             }
34387         }
34388         startNode.cascade(f);
34389         return r;
34390     },
34391
34392     /**
34393      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34394      * @param {String} path
34395      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34396      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34397      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34398      */
34399     expandPath : function(path, attr, callback){
34400         attr = attr || "id";
34401         var keys = path.split(this.pathSeparator);
34402         var curNode = this.root;
34403         if(curNode.attributes[attr] != keys[1]){ // invalid root
34404             if(callback){
34405                 callback(false, null);
34406             }
34407             return;
34408         }
34409         var index = 1;
34410         var f = function(){
34411             if(++index == keys.length){
34412                 if(callback){
34413                     callback(true, curNode);
34414                 }
34415                 return;
34416             }
34417             var c = curNode.findChild(attr, keys[index]);
34418             if(!c){
34419                 if(callback){
34420                     callback(false, curNode);
34421                 }
34422                 return;
34423             }
34424             curNode = c;
34425             c.expand(false, false, f);
34426         };
34427         curNode.expand(false, false, f);
34428     },
34429
34430     /**
34431      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34432      * @param {String} path
34433      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34434      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34435      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34436      */
34437     selectPath : function(path, attr, callback){
34438         attr = attr || "id";
34439         var keys = path.split(this.pathSeparator);
34440         var v = keys.pop();
34441         if(keys.length > 0){
34442             var f = function(success, node){
34443                 if(success && node){
34444                     var n = node.findChild(attr, v);
34445                     if(n){
34446                         n.select();
34447                         if(callback){
34448                             callback(true, n);
34449                         }
34450                     }else if(callback){
34451                         callback(false, n);
34452                     }
34453                 }else{
34454                     if(callback){
34455                         callback(false, n);
34456                     }
34457                 }
34458             };
34459             this.expandPath(keys.join(this.pathSeparator), attr, f);
34460         }else{
34461             this.root.select();
34462             if(callback){
34463                 callback(true, this.root);
34464             }
34465         }
34466     },
34467
34468     getTreeEl : function(){
34469         return this.el;
34470     },
34471
34472     /**
34473      * Trigger rendering of this TreePanel
34474      */
34475     render : function(){
34476         if (this.innerCt) {
34477             return this; // stop it rendering more than once!!
34478         }
34479         
34480         this.innerCt = this.el.createChild({tag:"ul",
34481                cls:"x-tree-root-ct " +
34482                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34483
34484         if(this.containerScroll){
34485             Roo.dd.ScrollManager.register(this.el);
34486         }
34487         if((this.enableDD || this.enableDrop) && !this.dropZone){
34488            /**
34489             * The dropZone used by this tree if drop is enabled
34490             * @type Roo.tree.TreeDropZone
34491             */
34492              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34493                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34494            });
34495         }
34496         if((this.enableDD || this.enableDrag) && !this.dragZone){
34497            /**
34498             * The dragZone used by this tree if drag is enabled
34499             * @type Roo.tree.TreeDragZone
34500             */
34501             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34502                ddGroup: this.ddGroup || "TreeDD",
34503                scroll: this.ddScroll
34504            });
34505         }
34506         this.getSelectionModel().init(this);
34507         if (!this.root) {
34508             Roo.log("ROOT not set in tree");
34509             return this;
34510         }
34511         this.root.render();
34512         if(!this.rootVisible){
34513             this.root.renderChildren();
34514         }
34515         return this;
34516     }
34517 });/*
34518  * Based on:
34519  * Ext JS Library 1.1.1
34520  * Copyright(c) 2006-2007, Ext JS, LLC.
34521  *
34522  * Originally Released Under LGPL - original licence link has changed is not relivant.
34523  *
34524  * Fork - LGPL
34525  * <script type="text/javascript">
34526  */
34527  
34528
34529 /**
34530  * @class Roo.tree.DefaultSelectionModel
34531  * @extends Roo.util.Observable
34532  * The default single selection for a TreePanel.
34533  * @param {Object} cfg Configuration
34534  */
34535 Roo.tree.DefaultSelectionModel = function(cfg){
34536    this.selNode = null;
34537    
34538    
34539    
34540    this.addEvents({
34541        /**
34542         * @event selectionchange
34543         * Fires when the selected node changes
34544         * @param {DefaultSelectionModel} this
34545         * @param {TreeNode} node the new selection
34546         */
34547        "selectionchange" : true,
34548
34549        /**
34550         * @event beforeselect
34551         * Fires before the selected node changes, return false to cancel the change
34552         * @param {DefaultSelectionModel} this
34553         * @param {TreeNode} node the new selection
34554         * @param {TreeNode} node the old selection
34555         */
34556        "beforeselect" : true
34557    });
34558    
34559     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34560 };
34561
34562 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34563     init : function(tree){
34564         this.tree = tree;
34565         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34566         tree.on("click", this.onNodeClick, this);
34567     },
34568     
34569     onNodeClick : function(node, e){
34570         if (e.ctrlKey && this.selNode == node)  {
34571             this.unselect(node);
34572             return;
34573         }
34574         this.select(node);
34575     },
34576     
34577     /**
34578      * Select a node.
34579      * @param {TreeNode} node The node to select
34580      * @return {TreeNode} The selected node
34581      */
34582     select : function(node){
34583         var last = this.selNode;
34584         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34585             if(last){
34586                 last.ui.onSelectedChange(false);
34587             }
34588             this.selNode = node;
34589             node.ui.onSelectedChange(true);
34590             this.fireEvent("selectionchange", this, node, last);
34591         }
34592         return node;
34593     },
34594     
34595     /**
34596      * Deselect a node.
34597      * @param {TreeNode} node The node to unselect
34598      */
34599     unselect : function(node){
34600         if(this.selNode == node){
34601             this.clearSelections();
34602         }    
34603     },
34604     
34605     /**
34606      * Clear all selections
34607      */
34608     clearSelections : function(){
34609         var n = this.selNode;
34610         if(n){
34611             n.ui.onSelectedChange(false);
34612             this.selNode = null;
34613             this.fireEvent("selectionchange", this, null);
34614         }
34615         return n;
34616     },
34617     
34618     /**
34619      * Get the selected node
34620      * @return {TreeNode} The selected node
34621      */
34622     getSelectedNode : function(){
34623         return this.selNode;    
34624     },
34625     
34626     /**
34627      * Returns true if the node is selected
34628      * @param {TreeNode} node The node to check
34629      * @return {Boolean}
34630      */
34631     isSelected : function(node){
34632         return this.selNode == node;  
34633     },
34634
34635     /**
34636      * Selects the node above the selected node in the tree, intelligently walking the nodes
34637      * @return TreeNode The new selection
34638      */
34639     selectPrevious : function(){
34640         var s = this.selNode || this.lastSelNode;
34641         if(!s){
34642             return null;
34643         }
34644         var ps = s.previousSibling;
34645         if(ps){
34646             if(!ps.isExpanded() || ps.childNodes.length < 1){
34647                 return this.select(ps);
34648             } else{
34649                 var lc = ps.lastChild;
34650                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34651                     lc = lc.lastChild;
34652                 }
34653                 return this.select(lc);
34654             }
34655         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34656             return this.select(s.parentNode);
34657         }
34658         return null;
34659     },
34660
34661     /**
34662      * Selects the node above the selected node in the tree, intelligently walking the nodes
34663      * @return TreeNode The new selection
34664      */
34665     selectNext : function(){
34666         var s = this.selNode || this.lastSelNode;
34667         if(!s){
34668             return null;
34669         }
34670         if(s.firstChild && s.isExpanded()){
34671              return this.select(s.firstChild);
34672          }else if(s.nextSibling){
34673              return this.select(s.nextSibling);
34674          }else if(s.parentNode){
34675             var newS = null;
34676             s.parentNode.bubble(function(){
34677                 if(this.nextSibling){
34678                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34679                     return false;
34680                 }
34681             });
34682             return newS;
34683          }
34684         return null;
34685     },
34686
34687     onKeyDown : function(e){
34688         var s = this.selNode || this.lastSelNode;
34689         // undesirable, but required
34690         var sm = this;
34691         if(!s){
34692             return;
34693         }
34694         var k = e.getKey();
34695         switch(k){
34696              case e.DOWN:
34697                  e.stopEvent();
34698                  this.selectNext();
34699              break;
34700              case e.UP:
34701                  e.stopEvent();
34702                  this.selectPrevious();
34703              break;
34704              case e.RIGHT:
34705                  e.preventDefault();
34706                  if(s.hasChildNodes()){
34707                      if(!s.isExpanded()){
34708                          s.expand();
34709                      }else if(s.firstChild){
34710                          this.select(s.firstChild, e);
34711                      }
34712                  }
34713              break;
34714              case e.LEFT:
34715                  e.preventDefault();
34716                  if(s.hasChildNodes() && s.isExpanded()){
34717                      s.collapse();
34718                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34719                      this.select(s.parentNode, e);
34720                  }
34721              break;
34722         };
34723     }
34724 });
34725
34726 /**
34727  * @class Roo.tree.MultiSelectionModel
34728  * @extends Roo.util.Observable
34729  * Multi selection for a TreePanel.
34730  * @param {Object} cfg Configuration
34731  */
34732 Roo.tree.MultiSelectionModel = function(){
34733    this.selNodes = [];
34734    this.selMap = {};
34735    this.addEvents({
34736        /**
34737         * @event selectionchange
34738         * Fires when the selected nodes change
34739         * @param {MultiSelectionModel} this
34740         * @param {Array} nodes Array of the selected nodes
34741         */
34742        "selectionchange" : true
34743    });
34744    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34745    
34746 };
34747
34748 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34749     init : function(tree){
34750         this.tree = tree;
34751         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34752         tree.on("click", this.onNodeClick, this);
34753     },
34754     
34755     onNodeClick : function(node, e){
34756         this.select(node, e, e.ctrlKey);
34757     },
34758     
34759     /**
34760      * Select a node.
34761      * @param {TreeNode} node The node to select
34762      * @param {EventObject} e (optional) An event associated with the selection
34763      * @param {Boolean} keepExisting True to retain existing selections
34764      * @return {TreeNode} The selected node
34765      */
34766     select : function(node, e, keepExisting){
34767         if(keepExisting !== true){
34768             this.clearSelections(true);
34769         }
34770         if(this.isSelected(node)){
34771             this.lastSelNode = node;
34772             return node;
34773         }
34774         this.selNodes.push(node);
34775         this.selMap[node.id] = node;
34776         this.lastSelNode = node;
34777         node.ui.onSelectedChange(true);
34778         this.fireEvent("selectionchange", this, this.selNodes);
34779         return node;
34780     },
34781     
34782     /**
34783      * Deselect a node.
34784      * @param {TreeNode} node The node to unselect
34785      */
34786     unselect : function(node){
34787         if(this.selMap[node.id]){
34788             node.ui.onSelectedChange(false);
34789             var sn = this.selNodes;
34790             var index = -1;
34791             if(sn.indexOf){
34792                 index = sn.indexOf(node);
34793             }else{
34794                 for(var i = 0, len = sn.length; i < len; i++){
34795                     if(sn[i] == node){
34796                         index = i;
34797                         break;
34798                     }
34799                 }
34800             }
34801             if(index != -1){
34802                 this.selNodes.splice(index, 1);
34803             }
34804             delete this.selMap[node.id];
34805             this.fireEvent("selectionchange", this, this.selNodes);
34806         }
34807     },
34808     
34809     /**
34810      * Clear all selections
34811      */
34812     clearSelections : function(suppressEvent){
34813         var sn = this.selNodes;
34814         if(sn.length > 0){
34815             for(var i = 0, len = sn.length; i < len; i++){
34816                 sn[i].ui.onSelectedChange(false);
34817             }
34818             this.selNodes = [];
34819             this.selMap = {};
34820             if(suppressEvent !== true){
34821                 this.fireEvent("selectionchange", this, this.selNodes);
34822             }
34823         }
34824     },
34825     
34826     /**
34827      * Returns true if the node is selected
34828      * @param {TreeNode} node The node to check
34829      * @return {Boolean}
34830      */
34831     isSelected : function(node){
34832         return this.selMap[node.id] ? true : false;  
34833     },
34834     
34835     /**
34836      * Returns an array of the selected nodes
34837      * @return {Array}
34838      */
34839     getSelectedNodes : function(){
34840         return this.selNodes;    
34841     },
34842
34843     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34844
34845     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34846
34847     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34848 });/*
34849  * Based on:
34850  * Ext JS Library 1.1.1
34851  * Copyright(c) 2006-2007, Ext JS, LLC.
34852  *
34853  * Originally Released Under LGPL - original licence link has changed is not relivant.
34854  *
34855  * Fork - LGPL
34856  * <script type="text/javascript">
34857  */
34858  
34859 /**
34860  * @class Roo.tree.TreeNode
34861  * @extends Roo.data.Node
34862  * @cfg {String} text The text for this node
34863  * @cfg {Boolean} expanded true to start the node expanded
34864  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34865  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34866  * @cfg {Boolean} disabled true to start the node disabled
34867  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34868  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34869  * @cfg {String} cls A css class to be added to the node
34870  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34871  * @cfg {String} href URL of the link used for the node (defaults to #)
34872  * @cfg {String} hrefTarget target frame for the link
34873  * @cfg {String} qtip An Ext QuickTip for the node
34874  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34875  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34876  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34877  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34878  * (defaults to undefined with no checkbox rendered)
34879  * @constructor
34880  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34881  */
34882 Roo.tree.TreeNode = function(attributes){
34883     attributes = attributes || {};
34884     if(typeof attributes == "string"){
34885         attributes = {text: attributes};
34886     }
34887     this.childrenRendered = false;
34888     this.rendered = false;
34889     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34890     this.expanded = attributes.expanded === true;
34891     this.isTarget = attributes.isTarget !== false;
34892     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34893     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34894
34895     /**
34896      * Read-only. The text for this node. To change it use setText().
34897      * @type String
34898      */
34899     this.text = attributes.text;
34900     /**
34901      * True if this node is disabled.
34902      * @type Boolean
34903      */
34904     this.disabled = attributes.disabled === true;
34905
34906     this.addEvents({
34907         /**
34908         * @event textchange
34909         * Fires when the text for this node is changed
34910         * @param {Node} this This node
34911         * @param {String} text The new text
34912         * @param {String} oldText The old text
34913         */
34914         "textchange" : true,
34915         /**
34916         * @event beforeexpand
34917         * Fires before this node is expanded, return false to cancel.
34918         * @param {Node} this This node
34919         * @param {Boolean} deep
34920         * @param {Boolean} anim
34921         */
34922         "beforeexpand" : true,
34923         /**
34924         * @event beforecollapse
34925         * Fires before this node is collapsed, return false to cancel.
34926         * @param {Node} this This node
34927         * @param {Boolean} deep
34928         * @param {Boolean} anim
34929         */
34930         "beforecollapse" : true,
34931         /**
34932         * @event expand
34933         * Fires when this node is expanded
34934         * @param {Node} this This node
34935         */
34936         "expand" : true,
34937         /**
34938         * @event disabledchange
34939         * Fires when the disabled status of this node changes
34940         * @param {Node} this This node
34941         * @param {Boolean} disabled
34942         */
34943         "disabledchange" : true,
34944         /**
34945         * @event collapse
34946         * Fires when this node is collapsed
34947         * @param {Node} this This node
34948         */
34949         "collapse" : true,
34950         /**
34951         * @event beforeclick
34952         * Fires before click processing. Return false to cancel the default action.
34953         * @param {Node} this This node
34954         * @param {Roo.EventObject} e The event object
34955         */
34956         "beforeclick":true,
34957         /**
34958         * @event checkchange
34959         * Fires when a node with a checkbox's checked property changes
34960         * @param {Node} this This node
34961         * @param {Boolean} checked
34962         */
34963         "checkchange":true,
34964         /**
34965         * @event click
34966         * Fires when this node is clicked
34967         * @param {Node} this This node
34968         * @param {Roo.EventObject} e The event object
34969         */
34970         "click":true,
34971         /**
34972         * @event dblclick
34973         * Fires when this node is double clicked
34974         * @param {Node} this This node
34975         * @param {Roo.EventObject} e The event object
34976         */
34977         "dblclick":true,
34978         /**
34979         * @event contextmenu
34980         * Fires when this node is right clicked
34981         * @param {Node} this This node
34982         * @param {Roo.EventObject} e The event object
34983         */
34984         "contextmenu":true,
34985         /**
34986         * @event beforechildrenrendered
34987         * Fires right before the child nodes for this node are rendered
34988         * @param {Node} this This node
34989         */
34990         "beforechildrenrendered":true
34991     });
34992
34993     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
34994
34995     /**
34996      * Read-only. The UI for this node
34997      * @type TreeNodeUI
34998      */
34999     this.ui = new uiClass(this);
35000     
35001     // finally support items[]
35002     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35003         return;
35004     }
35005     
35006     
35007     Roo.each(this.attributes.items, function(c) {
35008         this.appendChild(Roo.factory(c,Roo.Tree));
35009     }, this);
35010     delete this.attributes.items;
35011     
35012     
35013     
35014 };
35015 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35016     preventHScroll: true,
35017     /**
35018      * Returns true if this node is expanded
35019      * @return {Boolean}
35020      */
35021     isExpanded : function(){
35022         return this.expanded;
35023     },
35024
35025     /**
35026      * Returns the UI object for this node
35027      * @return {TreeNodeUI}
35028      */
35029     getUI : function(){
35030         return this.ui;
35031     },
35032
35033     // private override
35034     setFirstChild : function(node){
35035         var of = this.firstChild;
35036         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35037         if(this.childrenRendered && of && node != of){
35038             of.renderIndent(true, true);
35039         }
35040         if(this.rendered){
35041             this.renderIndent(true, true);
35042         }
35043     },
35044
35045     // private override
35046     setLastChild : function(node){
35047         var ol = this.lastChild;
35048         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35049         if(this.childrenRendered && ol && node != ol){
35050             ol.renderIndent(true, true);
35051         }
35052         if(this.rendered){
35053             this.renderIndent(true, true);
35054         }
35055     },
35056
35057     // these methods are overridden to provide lazy rendering support
35058     // private override
35059     appendChild : function()
35060     {
35061         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35062         if(node && this.childrenRendered){
35063             node.render();
35064         }
35065         this.ui.updateExpandIcon();
35066         return node;
35067     },
35068
35069     // private override
35070     removeChild : function(node){
35071         this.ownerTree.getSelectionModel().unselect(node);
35072         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35073         // if it's been rendered remove dom node
35074         if(this.childrenRendered){
35075             node.ui.remove();
35076         }
35077         if(this.childNodes.length < 1){
35078             this.collapse(false, false);
35079         }else{
35080             this.ui.updateExpandIcon();
35081         }
35082         if(!this.firstChild) {
35083             this.childrenRendered = false;
35084         }
35085         return node;
35086     },
35087
35088     // private override
35089     insertBefore : function(node, refNode){
35090         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35091         if(newNode && refNode && this.childrenRendered){
35092             node.render();
35093         }
35094         this.ui.updateExpandIcon();
35095         return newNode;
35096     },
35097
35098     /**
35099      * Sets the text for this node
35100      * @param {String} text
35101      */
35102     setText : function(text){
35103         var oldText = this.text;
35104         this.text = text;
35105         this.attributes.text = text;
35106         if(this.rendered){ // event without subscribing
35107             this.ui.onTextChange(this, text, oldText);
35108         }
35109         this.fireEvent("textchange", this, text, oldText);
35110     },
35111
35112     /**
35113      * Triggers selection of this node
35114      */
35115     select : function(){
35116         this.getOwnerTree().getSelectionModel().select(this);
35117     },
35118
35119     /**
35120      * Triggers deselection of this node
35121      */
35122     unselect : function(){
35123         this.getOwnerTree().getSelectionModel().unselect(this);
35124     },
35125
35126     /**
35127      * Returns true if this node is selected
35128      * @return {Boolean}
35129      */
35130     isSelected : function(){
35131         return this.getOwnerTree().getSelectionModel().isSelected(this);
35132     },
35133
35134     /**
35135      * Expand this node.
35136      * @param {Boolean} deep (optional) True to expand all children as well
35137      * @param {Boolean} anim (optional) false to cancel the default animation
35138      * @param {Function} callback (optional) A callback to be called when
35139      * expanding this node completes (does not wait for deep expand to complete).
35140      * Called with 1 parameter, this node.
35141      */
35142     expand : function(deep, anim, callback){
35143         if(!this.expanded){
35144             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35145                 return;
35146             }
35147             if(!this.childrenRendered){
35148                 this.renderChildren();
35149             }
35150             this.expanded = true;
35151             
35152             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35153                 this.ui.animExpand(function(){
35154                     this.fireEvent("expand", this);
35155                     if(typeof callback == "function"){
35156                         callback(this);
35157                     }
35158                     if(deep === true){
35159                         this.expandChildNodes(true);
35160                     }
35161                 }.createDelegate(this));
35162                 return;
35163             }else{
35164                 this.ui.expand();
35165                 this.fireEvent("expand", this);
35166                 if(typeof callback == "function"){
35167                     callback(this);
35168                 }
35169             }
35170         }else{
35171            if(typeof callback == "function"){
35172                callback(this);
35173            }
35174         }
35175         if(deep === true){
35176             this.expandChildNodes(true);
35177         }
35178     },
35179
35180     isHiddenRoot : function(){
35181         return this.isRoot && !this.getOwnerTree().rootVisible;
35182     },
35183
35184     /**
35185      * Collapse this node.
35186      * @param {Boolean} deep (optional) True to collapse all children as well
35187      * @param {Boolean} anim (optional) false to cancel the default animation
35188      */
35189     collapse : function(deep, anim){
35190         if(this.expanded && !this.isHiddenRoot()){
35191             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35192                 return;
35193             }
35194             this.expanded = false;
35195             if((this.getOwnerTree().animate && anim !== false) || anim){
35196                 this.ui.animCollapse(function(){
35197                     this.fireEvent("collapse", this);
35198                     if(deep === true){
35199                         this.collapseChildNodes(true);
35200                     }
35201                 }.createDelegate(this));
35202                 return;
35203             }else{
35204                 this.ui.collapse();
35205                 this.fireEvent("collapse", this);
35206             }
35207         }
35208         if(deep === true){
35209             var cs = this.childNodes;
35210             for(var i = 0, len = cs.length; i < len; i++) {
35211                 cs[i].collapse(true, false);
35212             }
35213         }
35214     },
35215
35216     // private
35217     delayedExpand : function(delay){
35218         if(!this.expandProcId){
35219             this.expandProcId = this.expand.defer(delay, this);
35220         }
35221     },
35222
35223     // private
35224     cancelExpand : function(){
35225         if(this.expandProcId){
35226             clearTimeout(this.expandProcId);
35227         }
35228         this.expandProcId = false;
35229     },
35230
35231     /**
35232      * Toggles expanded/collapsed state of the node
35233      */
35234     toggle : function(){
35235         if(this.expanded){
35236             this.collapse();
35237         }else{
35238             this.expand();
35239         }
35240     },
35241
35242     /**
35243      * Ensures all parent nodes are expanded
35244      */
35245     ensureVisible : function(callback){
35246         var tree = this.getOwnerTree();
35247         tree.expandPath(this.parentNode.getPath(), false, function(){
35248             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35249             Roo.callback(callback);
35250         }.createDelegate(this));
35251     },
35252
35253     /**
35254      * Expand all child nodes
35255      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35256      */
35257     expandChildNodes : function(deep){
35258         var cs = this.childNodes;
35259         for(var i = 0, len = cs.length; i < len; i++) {
35260                 cs[i].expand(deep);
35261         }
35262     },
35263
35264     /**
35265      * Collapse all child nodes
35266      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35267      */
35268     collapseChildNodes : function(deep){
35269         var cs = this.childNodes;
35270         for(var i = 0, len = cs.length; i < len; i++) {
35271                 cs[i].collapse(deep);
35272         }
35273     },
35274
35275     /**
35276      * Disables this node
35277      */
35278     disable : function(){
35279         this.disabled = true;
35280         this.unselect();
35281         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35282             this.ui.onDisableChange(this, true);
35283         }
35284         this.fireEvent("disabledchange", this, true);
35285     },
35286
35287     /**
35288      * Enables this node
35289      */
35290     enable : function(){
35291         this.disabled = false;
35292         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35293             this.ui.onDisableChange(this, false);
35294         }
35295         this.fireEvent("disabledchange", this, false);
35296     },
35297
35298     // private
35299     renderChildren : function(suppressEvent){
35300         if(suppressEvent !== false){
35301             this.fireEvent("beforechildrenrendered", this);
35302         }
35303         var cs = this.childNodes;
35304         for(var i = 0, len = cs.length; i < len; i++){
35305             cs[i].render(true);
35306         }
35307         this.childrenRendered = true;
35308     },
35309
35310     // private
35311     sort : function(fn, scope){
35312         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35313         if(this.childrenRendered){
35314             var cs = this.childNodes;
35315             for(var i = 0, len = cs.length; i < len; i++){
35316                 cs[i].render(true);
35317             }
35318         }
35319     },
35320
35321     // private
35322     render : function(bulkRender){
35323         this.ui.render(bulkRender);
35324         if(!this.rendered){
35325             this.rendered = true;
35326             if(this.expanded){
35327                 this.expanded = false;
35328                 this.expand(false, false);
35329             }
35330         }
35331     },
35332
35333     // private
35334     renderIndent : function(deep, refresh){
35335         if(refresh){
35336             this.ui.childIndent = null;
35337         }
35338         this.ui.renderIndent();
35339         if(deep === true && this.childrenRendered){
35340             var cs = this.childNodes;
35341             for(var i = 0, len = cs.length; i < len; i++){
35342                 cs[i].renderIndent(true, refresh);
35343             }
35344         }
35345     }
35346 });/*
35347  * Based on:
35348  * Ext JS Library 1.1.1
35349  * Copyright(c) 2006-2007, Ext JS, LLC.
35350  *
35351  * Originally Released Under LGPL - original licence link has changed is not relivant.
35352  *
35353  * Fork - LGPL
35354  * <script type="text/javascript">
35355  */
35356  
35357 /**
35358  * @class Roo.tree.AsyncTreeNode
35359  * @extends Roo.tree.TreeNode
35360  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35361  * @constructor
35362  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35363  */
35364  Roo.tree.AsyncTreeNode = function(config){
35365     this.loaded = false;
35366     this.loading = false;
35367     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35368     /**
35369     * @event beforeload
35370     * Fires before this node is loaded, return false to cancel
35371     * @param {Node} this This node
35372     */
35373     this.addEvents({'beforeload':true, 'load': true});
35374     /**
35375     * @event load
35376     * Fires when this node is loaded
35377     * @param {Node} this This node
35378     */
35379     /**
35380      * The loader used by this node (defaults to using the tree's defined loader)
35381      * @type TreeLoader
35382      * @property loader
35383      */
35384 };
35385 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35386     expand : function(deep, anim, callback){
35387         if(this.loading){ // if an async load is already running, waiting til it's done
35388             var timer;
35389             var f = function(){
35390                 if(!this.loading){ // done loading
35391                     clearInterval(timer);
35392                     this.expand(deep, anim, callback);
35393                 }
35394             }.createDelegate(this);
35395             timer = setInterval(f, 200);
35396             return;
35397         }
35398         if(!this.loaded){
35399             if(this.fireEvent("beforeload", this) === false){
35400                 return;
35401             }
35402             this.loading = true;
35403             this.ui.beforeLoad(this);
35404             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35405             if(loader){
35406                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35407                 return;
35408             }
35409         }
35410         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35411     },
35412     
35413     /**
35414      * Returns true if this node is currently loading
35415      * @return {Boolean}
35416      */
35417     isLoading : function(){
35418         return this.loading;  
35419     },
35420     
35421     loadComplete : function(deep, anim, callback){
35422         this.loading = false;
35423         this.loaded = true;
35424         this.ui.afterLoad(this);
35425         this.fireEvent("load", this);
35426         this.expand(deep, anim, callback);
35427     },
35428     
35429     /**
35430      * Returns true if this node has been loaded
35431      * @return {Boolean}
35432      */
35433     isLoaded : function(){
35434         return this.loaded;
35435     },
35436     
35437     hasChildNodes : function(){
35438         if(!this.isLeaf() && !this.loaded){
35439             return true;
35440         }else{
35441             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35442         }
35443     },
35444
35445     /**
35446      * Trigger a reload for this node
35447      * @param {Function} callback
35448      */
35449     reload : function(callback){
35450         this.collapse(false, false);
35451         while(this.firstChild){
35452             this.removeChild(this.firstChild);
35453         }
35454         this.childrenRendered = false;
35455         this.loaded = false;
35456         if(this.isHiddenRoot()){
35457             this.expanded = false;
35458         }
35459         this.expand(false, false, callback);
35460     }
35461 });/*
35462  * Based on:
35463  * Ext JS Library 1.1.1
35464  * Copyright(c) 2006-2007, Ext JS, LLC.
35465  *
35466  * Originally Released Under LGPL - original licence link has changed is not relivant.
35467  *
35468  * Fork - LGPL
35469  * <script type="text/javascript">
35470  */
35471  
35472 /**
35473  * @class Roo.tree.TreeNodeUI
35474  * @constructor
35475  * @param {Object} node The node to render
35476  * The TreeNode UI implementation is separate from the
35477  * tree implementation. Unless you are customizing the tree UI,
35478  * you should never have to use this directly.
35479  */
35480 Roo.tree.TreeNodeUI = function(node){
35481     this.node = node;
35482     this.rendered = false;
35483     this.animating = false;
35484     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35485 };
35486
35487 Roo.tree.TreeNodeUI.prototype = {
35488     removeChild : function(node){
35489         if(this.rendered){
35490             this.ctNode.removeChild(node.ui.getEl());
35491         }
35492     },
35493
35494     beforeLoad : function(){
35495          this.addClass("x-tree-node-loading");
35496     },
35497
35498     afterLoad : function(){
35499          this.removeClass("x-tree-node-loading");
35500     },
35501
35502     onTextChange : function(node, text, oldText){
35503         if(this.rendered){
35504             this.textNode.innerHTML = text;
35505         }
35506     },
35507
35508     onDisableChange : function(node, state){
35509         this.disabled = state;
35510         if(state){
35511             this.addClass("x-tree-node-disabled");
35512         }else{
35513             this.removeClass("x-tree-node-disabled");
35514         }
35515     },
35516
35517     onSelectedChange : function(state){
35518         if(state){
35519             this.focus();
35520             this.addClass("x-tree-selected");
35521         }else{
35522             //this.blur();
35523             this.removeClass("x-tree-selected");
35524         }
35525     },
35526
35527     onMove : function(tree, node, oldParent, newParent, index, refNode){
35528         this.childIndent = null;
35529         if(this.rendered){
35530             var targetNode = newParent.ui.getContainer();
35531             if(!targetNode){//target not rendered
35532                 this.holder = document.createElement("div");
35533                 this.holder.appendChild(this.wrap);
35534                 return;
35535             }
35536             var insertBefore = refNode ? refNode.ui.getEl() : null;
35537             if(insertBefore){
35538                 targetNode.insertBefore(this.wrap, insertBefore);
35539             }else{
35540                 targetNode.appendChild(this.wrap);
35541             }
35542             this.node.renderIndent(true);
35543         }
35544     },
35545
35546     addClass : function(cls){
35547         if(this.elNode){
35548             Roo.fly(this.elNode).addClass(cls);
35549         }
35550     },
35551
35552     removeClass : function(cls){
35553         if(this.elNode){
35554             Roo.fly(this.elNode).removeClass(cls);
35555         }
35556     },
35557
35558     remove : function(){
35559         if(this.rendered){
35560             this.holder = document.createElement("div");
35561             this.holder.appendChild(this.wrap);
35562         }
35563     },
35564
35565     fireEvent : function(){
35566         return this.node.fireEvent.apply(this.node, arguments);
35567     },
35568
35569     initEvents : function(){
35570         this.node.on("move", this.onMove, this);
35571         var E = Roo.EventManager;
35572         var a = this.anchor;
35573
35574         var el = Roo.fly(a, '_treeui');
35575
35576         if(Roo.isOpera){ // opera render bug ignores the CSS
35577             el.setStyle("text-decoration", "none");
35578         }
35579
35580         el.on("click", this.onClick, this);
35581         el.on("dblclick", this.onDblClick, this);
35582
35583         if(this.checkbox){
35584             Roo.EventManager.on(this.checkbox,
35585                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35586         }
35587
35588         el.on("contextmenu", this.onContextMenu, this);
35589
35590         var icon = Roo.fly(this.iconNode);
35591         icon.on("click", this.onClick, this);
35592         icon.on("dblclick", this.onDblClick, this);
35593         icon.on("contextmenu", this.onContextMenu, this);
35594         E.on(this.ecNode, "click", this.ecClick, this, true);
35595
35596         if(this.node.disabled){
35597             this.addClass("x-tree-node-disabled");
35598         }
35599         if(this.node.hidden){
35600             this.addClass("x-tree-node-disabled");
35601         }
35602         var ot = this.node.getOwnerTree();
35603         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35604         if(dd && (!this.node.isRoot || ot.rootVisible)){
35605             Roo.dd.Registry.register(this.elNode, {
35606                 node: this.node,
35607                 handles: this.getDDHandles(),
35608                 isHandle: false
35609             });
35610         }
35611     },
35612
35613     getDDHandles : function(){
35614         return [this.iconNode, this.textNode];
35615     },
35616
35617     hide : function(){
35618         if(this.rendered){
35619             this.wrap.style.display = "none";
35620         }
35621     },
35622
35623     show : function(){
35624         if(this.rendered){
35625             this.wrap.style.display = "";
35626         }
35627     },
35628
35629     onContextMenu : function(e){
35630         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35631             e.preventDefault();
35632             this.focus();
35633             this.fireEvent("contextmenu", this.node, e);
35634         }
35635     },
35636
35637     onClick : function(e){
35638         if(this.dropping){
35639             e.stopEvent();
35640             return;
35641         }
35642         if(this.fireEvent("beforeclick", this.node, e) !== false){
35643             if(!this.disabled && this.node.attributes.href){
35644                 this.fireEvent("click", this.node, e);
35645                 return;
35646             }
35647             e.preventDefault();
35648             if(this.disabled){
35649                 return;
35650             }
35651
35652             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35653                 this.node.toggle();
35654             }
35655
35656             this.fireEvent("click", this.node, e);
35657         }else{
35658             e.stopEvent();
35659         }
35660     },
35661
35662     onDblClick : function(e){
35663         e.preventDefault();
35664         if(this.disabled){
35665             return;
35666         }
35667         if(this.checkbox){
35668             this.toggleCheck();
35669         }
35670         if(!this.animating && this.node.hasChildNodes()){
35671             this.node.toggle();
35672         }
35673         this.fireEvent("dblclick", this.node, e);
35674     },
35675
35676     onCheckChange : function(){
35677         var checked = this.checkbox.checked;
35678         this.node.attributes.checked = checked;
35679         this.fireEvent('checkchange', this.node, checked);
35680     },
35681
35682     ecClick : function(e){
35683         if(!this.animating && this.node.hasChildNodes()){
35684             this.node.toggle();
35685         }
35686     },
35687
35688     startDrop : function(){
35689         this.dropping = true;
35690     },
35691
35692     // delayed drop so the click event doesn't get fired on a drop
35693     endDrop : function(){
35694        setTimeout(function(){
35695            this.dropping = false;
35696        }.createDelegate(this), 50);
35697     },
35698
35699     expand : function(){
35700         this.updateExpandIcon();
35701         this.ctNode.style.display = "";
35702     },
35703
35704     focus : function(){
35705         if(!this.node.preventHScroll){
35706             try{this.anchor.focus();
35707             }catch(e){}
35708         }else if(!Roo.isIE){
35709             try{
35710                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35711                 var l = noscroll.scrollLeft;
35712                 this.anchor.focus();
35713                 noscroll.scrollLeft = l;
35714             }catch(e){}
35715         }
35716     },
35717
35718     toggleCheck : function(value){
35719         var cb = this.checkbox;
35720         if(cb){
35721             cb.checked = (value === undefined ? !cb.checked : value);
35722         }
35723     },
35724
35725     blur : function(){
35726         try{
35727             this.anchor.blur();
35728         }catch(e){}
35729     },
35730
35731     animExpand : function(callback){
35732         var ct = Roo.get(this.ctNode);
35733         ct.stopFx();
35734         if(!this.node.hasChildNodes()){
35735             this.updateExpandIcon();
35736             this.ctNode.style.display = "";
35737             Roo.callback(callback);
35738             return;
35739         }
35740         this.animating = true;
35741         this.updateExpandIcon();
35742
35743         ct.slideIn('t', {
35744            callback : function(){
35745                this.animating = false;
35746                Roo.callback(callback);
35747             },
35748             scope: this,
35749             duration: this.node.ownerTree.duration || .25
35750         });
35751     },
35752
35753     highlight : function(){
35754         var tree = this.node.getOwnerTree();
35755         Roo.fly(this.wrap).highlight(
35756             tree.hlColor || "C3DAF9",
35757             {endColor: tree.hlBaseColor}
35758         );
35759     },
35760
35761     collapse : function(){
35762         this.updateExpandIcon();
35763         this.ctNode.style.display = "none";
35764     },
35765
35766     animCollapse : function(callback){
35767         var ct = Roo.get(this.ctNode);
35768         ct.enableDisplayMode('block');
35769         ct.stopFx();
35770
35771         this.animating = true;
35772         this.updateExpandIcon();
35773
35774         ct.slideOut('t', {
35775             callback : function(){
35776                this.animating = false;
35777                Roo.callback(callback);
35778             },
35779             scope: this,
35780             duration: this.node.ownerTree.duration || .25
35781         });
35782     },
35783
35784     getContainer : function(){
35785         return this.ctNode;
35786     },
35787
35788     getEl : function(){
35789         return this.wrap;
35790     },
35791
35792     appendDDGhost : function(ghostNode){
35793         ghostNode.appendChild(this.elNode.cloneNode(true));
35794     },
35795
35796     getDDRepairXY : function(){
35797         return Roo.lib.Dom.getXY(this.iconNode);
35798     },
35799
35800     onRender : function(){
35801         this.render();
35802     },
35803
35804     render : function(bulkRender){
35805         var n = this.node, a = n.attributes;
35806         var targetNode = n.parentNode ?
35807               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35808
35809         if(!this.rendered){
35810             this.rendered = true;
35811
35812             this.renderElements(n, a, targetNode, bulkRender);
35813
35814             if(a.qtip){
35815                if(this.textNode.setAttributeNS){
35816                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35817                    if(a.qtipTitle){
35818                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35819                    }
35820                }else{
35821                    this.textNode.setAttribute("ext:qtip", a.qtip);
35822                    if(a.qtipTitle){
35823                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35824                    }
35825                }
35826             }else if(a.qtipCfg){
35827                 a.qtipCfg.target = Roo.id(this.textNode);
35828                 Roo.QuickTips.register(a.qtipCfg);
35829             }
35830             this.initEvents();
35831             if(!this.node.expanded){
35832                 this.updateExpandIcon();
35833             }
35834         }else{
35835             if(bulkRender === true) {
35836                 targetNode.appendChild(this.wrap);
35837             }
35838         }
35839     },
35840
35841     renderElements : function(n, a, targetNode, bulkRender)
35842     {
35843         // add some indent caching, this helps performance when rendering a large tree
35844         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35845         var t = n.getOwnerTree();
35846         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35847         if (typeof(n.attributes.html) != 'undefined') {
35848             txt = n.attributes.html;
35849         }
35850         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35851         var cb = typeof a.checked == 'boolean';
35852         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35853         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35854             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35855             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35856             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35857             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35858             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35859              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35860                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35861             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35862             "</li>"];
35863
35864         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35865             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35866                                 n.nextSibling.ui.getEl(), buf.join(""));
35867         }else{
35868             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35869         }
35870
35871         this.elNode = this.wrap.childNodes[0];
35872         this.ctNode = this.wrap.childNodes[1];
35873         var cs = this.elNode.childNodes;
35874         this.indentNode = cs[0];
35875         this.ecNode = cs[1];
35876         this.iconNode = cs[2];
35877         var index = 3;
35878         if(cb){
35879             this.checkbox = cs[3];
35880             index++;
35881         }
35882         this.anchor = cs[index];
35883         this.textNode = cs[index].firstChild;
35884     },
35885
35886     getAnchor : function(){
35887         return this.anchor;
35888     },
35889
35890     getTextEl : function(){
35891         return this.textNode;
35892     },
35893
35894     getIconEl : function(){
35895         return this.iconNode;
35896     },
35897
35898     isChecked : function(){
35899         return this.checkbox ? this.checkbox.checked : false;
35900     },
35901
35902     updateExpandIcon : function(){
35903         if(this.rendered){
35904             var n = this.node, c1, c2;
35905             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35906             var hasChild = n.hasChildNodes();
35907             if(hasChild){
35908                 if(n.expanded){
35909                     cls += "-minus";
35910                     c1 = "x-tree-node-collapsed";
35911                     c2 = "x-tree-node-expanded";
35912                 }else{
35913                     cls += "-plus";
35914                     c1 = "x-tree-node-expanded";
35915                     c2 = "x-tree-node-collapsed";
35916                 }
35917                 if(this.wasLeaf){
35918                     this.removeClass("x-tree-node-leaf");
35919                     this.wasLeaf = false;
35920                 }
35921                 if(this.c1 != c1 || this.c2 != c2){
35922                     Roo.fly(this.elNode).replaceClass(c1, c2);
35923                     this.c1 = c1; this.c2 = c2;
35924                 }
35925             }else{
35926                 // this changes non-leafs into leafs if they have no children.
35927                 // it's not very rational behaviour..
35928                 
35929                 if(!this.wasLeaf && this.node.leaf){
35930                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35931                     delete this.c1;
35932                     delete this.c2;
35933                     this.wasLeaf = true;
35934                 }
35935             }
35936             var ecc = "x-tree-ec-icon "+cls;
35937             if(this.ecc != ecc){
35938                 this.ecNode.className = ecc;
35939                 this.ecc = ecc;
35940             }
35941         }
35942     },
35943
35944     getChildIndent : function(){
35945         if(!this.childIndent){
35946             var buf = [];
35947             var p = this.node;
35948             while(p){
35949                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35950                     if(!p.isLast()) {
35951                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35952                     } else {
35953                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35954                     }
35955                 }
35956                 p = p.parentNode;
35957             }
35958             this.childIndent = buf.join("");
35959         }
35960         return this.childIndent;
35961     },
35962
35963     renderIndent : function(){
35964         if(this.rendered){
35965             var indent = "";
35966             var p = this.node.parentNode;
35967             if(p){
35968                 indent = p.ui.getChildIndent();
35969             }
35970             if(this.indentMarkup != indent){ // don't rerender if not required
35971                 this.indentNode.innerHTML = indent;
35972                 this.indentMarkup = indent;
35973             }
35974             this.updateExpandIcon();
35975         }
35976     }
35977 };
35978
35979 Roo.tree.RootTreeNodeUI = function(){
35980     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
35981 };
35982 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
35983     render : function(){
35984         if(!this.rendered){
35985             var targetNode = this.node.ownerTree.innerCt.dom;
35986             this.node.expanded = true;
35987             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
35988             this.wrap = this.ctNode = targetNode.firstChild;
35989         }
35990     },
35991     collapse : function(){
35992     },
35993     expand : function(){
35994     }
35995 });/*
35996  * Based on:
35997  * Ext JS Library 1.1.1
35998  * Copyright(c) 2006-2007, Ext JS, LLC.
35999  *
36000  * Originally Released Under LGPL - original licence link has changed is not relivant.
36001  *
36002  * Fork - LGPL
36003  * <script type="text/javascript">
36004  */
36005 /**
36006  * @class Roo.tree.TreeLoader
36007  * @extends Roo.util.Observable
36008  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36009  * nodes from a specified URL. The response must be a javascript Array definition
36010  * who's elements are node definition objects. eg:
36011  * <pre><code>
36012 {  success : true,
36013    data :      [
36014    
36015     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36016     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36017     ]
36018 }
36019
36020
36021 </code></pre>
36022  * <br><br>
36023  * The old style respose with just an array is still supported, but not recommended.
36024  * <br><br>
36025  *
36026  * A server request is sent, and child nodes are loaded only when a node is expanded.
36027  * The loading node's id is passed to the server under the parameter name "node" to
36028  * enable the server to produce the correct child nodes.
36029  * <br><br>
36030  * To pass extra parameters, an event handler may be attached to the "beforeload"
36031  * event, and the parameters specified in the TreeLoader's baseParams property:
36032  * <pre><code>
36033     myTreeLoader.on("beforeload", function(treeLoader, node) {
36034         this.baseParams.category = node.attributes.category;
36035     }, this);
36036     
36037 </code></pre>
36038  *
36039  * This would pass an HTTP parameter called "category" to the server containing
36040  * the value of the Node's "category" attribute.
36041  * @constructor
36042  * Creates a new Treeloader.
36043  * @param {Object} config A config object containing config properties.
36044  */
36045 Roo.tree.TreeLoader = function(config){
36046     this.baseParams = {};
36047     this.requestMethod = "POST";
36048     Roo.apply(this, config);
36049
36050     this.addEvents({
36051     
36052         /**
36053          * @event beforeload
36054          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36055          * @param {Object} This TreeLoader object.
36056          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36057          * @param {Object} callback The callback function specified in the {@link #load} call.
36058          */
36059         beforeload : true,
36060         /**
36061          * @event load
36062          * Fires when the node has been successfuly loaded.
36063          * @param {Object} This TreeLoader object.
36064          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36065          * @param {Object} response The response object containing the data from the server.
36066          */
36067         load : true,
36068         /**
36069          * @event loadexception
36070          * Fires if the network request failed.
36071          * @param {Object} This TreeLoader object.
36072          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36073          * @param {Object} response The response object containing the data from the server.
36074          */
36075         loadexception : true,
36076         /**
36077          * @event create
36078          * Fires before a node is created, enabling you to return custom Node types 
36079          * @param {Object} This TreeLoader object.
36080          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36081          */
36082         create : true
36083     });
36084
36085     Roo.tree.TreeLoader.superclass.constructor.call(this);
36086 };
36087
36088 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36089     /**
36090     * @cfg {String} dataUrl The URL from which to request a Json string which
36091     * specifies an array of node definition object representing the child nodes
36092     * to be loaded.
36093     */
36094     /**
36095     * @cfg {String} requestMethod either GET or POST
36096     * defaults to POST (due to BC)
36097     * to be loaded.
36098     */
36099     /**
36100     * @cfg {Object} baseParams (optional) An object containing properties which
36101     * specify HTTP parameters to be passed to each request for child nodes.
36102     */
36103     /**
36104     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36105     * created by this loader. If the attributes sent by the server have an attribute in this object,
36106     * they take priority.
36107     */
36108     /**
36109     * @cfg {Object} uiProviders (optional) An object containing properties which
36110     * 
36111     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36112     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36113     * <i>uiProvider</i> attribute of a returned child node is a string rather
36114     * than a reference to a TreeNodeUI implementation, this that string value
36115     * is used as a property name in the uiProviders object. You can define the provider named
36116     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36117     */
36118     uiProviders : {},
36119
36120     /**
36121     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36122     * child nodes before loading.
36123     */
36124     clearOnLoad : true,
36125
36126     /**
36127     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36128     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36129     * Grid query { data : [ .....] }
36130     */
36131     
36132     root : false,
36133      /**
36134     * @cfg {String} queryParam (optional) 
36135     * Name of the query as it will be passed on the querystring (defaults to 'node')
36136     * eg. the request will be ?node=[id]
36137     */
36138     
36139     
36140     queryParam: false,
36141     
36142     /**
36143      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36144      * This is called automatically when a node is expanded, but may be used to reload
36145      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36146      * @param {Roo.tree.TreeNode} node
36147      * @param {Function} callback
36148      */
36149     load : function(node, callback){
36150         if(this.clearOnLoad){
36151             while(node.firstChild){
36152                 node.removeChild(node.firstChild);
36153             }
36154         }
36155         if(node.attributes.children){ // preloaded json children
36156             var cs = node.attributes.children;
36157             for(var i = 0, len = cs.length; i < len; i++){
36158                 node.appendChild(this.createNode(cs[i]));
36159             }
36160             if(typeof callback == "function"){
36161                 callback();
36162             }
36163         }else if(this.dataUrl){
36164             this.requestData(node, callback);
36165         }
36166     },
36167
36168     getParams: function(node){
36169         var buf = [], bp = this.baseParams;
36170         for(var key in bp){
36171             if(typeof bp[key] != "function"){
36172                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36173             }
36174         }
36175         var n = this.queryParam === false ? 'node' : this.queryParam;
36176         buf.push(n + "=", encodeURIComponent(node.id));
36177         return buf.join("");
36178     },
36179
36180     requestData : function(node, callback){
36181         if(this.fireEvent("beforeload", this, node, callback) !== false){
36182             this.transId = Roo.Ajax.request({
36183                 method:this.requestMethod,
36184                 url: this.dataUrl||this.url,
36185                 success: this.handleResponse,
36186                 failure: this.handleFailure,
36187                 scope: this,
36188                 argument: {callback: callback, node: node},
36189                 params: this.getParams(node)
36190             });
36191         }else{
36192             // if the load is cancelled, make sure we notify
36193             // the node that we are done
36194             if(typeof callback == "function"){
36195                 callback();
36196             }
36197         }
36198     },
36199
36200     isLoading : function(){
36201         return this.transId ? true : false;
36202     },
36203
36204     abort : function(){
36205         if(this.isLoading()){
36206             Roo.Ajax.abort(this.transId);
36207         }
36208     },
36209
36210     // private
36211     createNode : function(attr)
36212     {
36213         // apply baseAttrs, nice idea Corey!
36214         if(this.baseAttrs){
36215             Roo.applyIf(attr, this.baseAttrs);
36216         }
36217         if(this.applyLoader !== false){
36218             attr.loader = this;
36219         }
36220         // uiProvider = depreciated..
36221         
36222         if(typeof(attr.uiProvider) == 'string'){
36223            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36224                 /**  eval:var:attr */ eval(attr.uiProvider);
36225         }
36226         if(typeof(this.uiProviders['default']) != 'undefined') {
36227             attr.uiProvider = this.uiProviders['default'];
36228         }
36229         
36230         this.fireEvent('create', this, attr);
36231         
36232         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36233         return(attr.leaf ?
36234                         new Roo.tree.TreeNode(attr) :
36235                         new Roo.tree.AsyncTreeNode(attr));
36236     },
36237
36238     processResponse : function(response, node, callback)
36239     {
36240         var json = response.responseText;
36241         try {
36242             
36243             var o = Roo.decode(json);
36244             
36245             if (this.root === false && typeof(o.success) != undefined) {
36246                 this.root = 'data'; // the default behaviour for list like data..
36247                 }
36248                 
36249             if (this.root !== false &&  !o.success) {
36250                 // it's a failure condition.
36251                 var a = response.argument;
36252                 this.fireEvent("loadexception", this, a.node, response);
36253                 Roo.log("Load failed - should have a handler really");
36254                 return;
36255             }
36256             
36257             
36258             
36259             if (this.root !== false) {
36260                  o = o[this.root];
36261             }
36262             
36263             for(var i = 0, len = o.length; i < len; i++){
36264                 var n = this.createNode(o[i]);
36265                 if(n){
36266                     node.appendChild(n);
36267                 }
36268             }
36269             if(typeof callback == "function"){
36270                 callback(this, node);
36271             }
36272         }catch(e){
36273             this.handleFailure(response);
36274         }
36275     },
36276
36277     handleResponse : function(response){
36278         this.transId = false;
36279         var a = response.argument;
36280         this.processResponse(response, a.node, a.callback);
36281         this.fireEvent("load", this, a.node, response);
36282     },
36283
36284     handleFailure : function(response)
36285     {
36286         // should handle failure better..
36287         this.transId = false;
36288         var a = response.argument;
36289         this.fireEvent("loadexception", this, a.node, response);
36290         if(typeof a.callback == "function"){
36291             a.callback(this, a.node);
36292         }
36293     }
36294 });/*
36295  * Based on:
36296  * Ext JS Library 1.1.1
36297  * Copyright(c) 2006-2007, Ext JS, LLC.
36298  *
36299  * Originally Released Under LGPL - original licence link has changed is not relivant.
36300  *
36301  * Fork - LGPL
36302  * <script type="text/javascript">
36303  */
36304
36305 /**
36306 * @class Roo.tree.TreeFilter
36307 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36308 * @param {TreePanel} tree
36309 * @param {Object} config (optional)
36310  */
36311 Roo.tree.TreeFilter = function(tree, config){
36312     this.tree = tree;
36313     this.filtered = {};
36314     Roo.apply(this, config);
36315 };
36316
36317 Roo.tree.TreeFilter.prototype = {
36318     clearBlank:false,
36319     reverse:false,
36320     autoClear:false,
36321     remove:false,
36322
36323      /**
36324      * Filter the data by a specific attribute.
36325      * @param {String/RegExp} value Either string that the attribute value
36326      * should start with or a RegExp to test against the attribute
36327      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36328      * @param {TreeNode} startNode (optional) The node to start the filter at.
36329      */
36330     filter : function(value, attr, startNode){
36331         attr = attr || "text";
36332         var f;
36333         if(typeof value == "string"){
36334             var vlen = value.length;
36335             // auto clear empty filter
36336             if(vlen == 0 && this.clearBlank){
36337                 this.clear();
36338                 return;
36339             }
36340             value = value.toLowerCase();
36341             f = function(n){
36342                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36343             };
36344         }else if(value.exec){ // regex?
36345             f = function(n){
36346                 return value.test(n.attributes[attr]);
36347             };
36348         }else{
36349             throw 'Illegal filter type, must be string or regex';
36350         }
36351         this.filterBy(f, null, startNode);
36352         },
36353
36354     /**
36355      * Filter by a function. The passed function will be called with each
36356      * node in the tree (or from the startNode). If the function returns true, the node is kept
36357      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36358      * @param {Function} fn The filter function
36359      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36360      */
36361     filterBy : function(fn, scope, startNode){
36362         startNode = startNode || this.tree.root;
36363         if(this.autoClear){
36364             this.clear();
36365         }
36366         var af = this.filtered, rv = this.reverse;
36367         var f = function(n){
36368             if(n == startNode){
36369                 return true;
36370             }
36371             if(af[n.id]){
36372                 return false;
36373             }
36374             var m = fn.call(scope || n, n);
36375             if(!m || rv){
36376                 af[n.id] = n;
36377                 n.ui.hide();
36378                 return false;
36379             }
36380             return true;
36381         };
36382         startNode.cascade(f);
36383         if(this.remove){
36384            for(var id in af){
36385                if(typeof id != "function"){
36386                    var n = af[id];
36387                    if(n && n.parentNode){
36388                        n.parentNode.removeChild(n);
36389                    }
36390                }
36391            }
36392         }
36393     },
36394
36395     /**
36396      * Clears the current filter. Note: with the "remove" option
36397      * set a filter cannot be cleared.
36398      */
36399     clear : function(){
36400         var t = this.tree;
36401         var af = this.filtered;
36402         for(var id in af){
36403             if(typeof id != "function"){
36404                 var n = af[id];
36405                 if(n){
36406                     n.ui.show();
36407                 }
36408             }
36409         }
36410         this.filtered = {};
36411     }
36412 };
36413 /*
36414  * Based on:
36415  * Ext JS Library 1.1.1
36416  * Copyright(c) 2006-2007, Ext JS, LLC.
36417  *
36418  * Originally Released Under LGPL - original licence link has changed is not relivant.
36419  *
36420  * Fork - LGPL
36421  * <script type="text/javascript">
36422  */
36423  
36424
36425 /**
36426  * @class Roo.tree.TreeSorter
36427  * Provides sorting of nodes in a TreePanel
36428  * 
36429  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36430  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36431  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36432  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36433  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36434  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36435  * @constructor
36436  * @param {TreePanel} tree
36437  * @param {Object} config
36438  */
36439 Roo.tree.TreeSorter = function(tree, config){
36440     Roo.apply(this, config);
36441     tree.on("beforechildrenrendered", this.doSort, this);
36442     tree.on("append", this.updateSort, this);
36443     tree.on("insert", this.updateSort, this);
36444     
36445     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36446     var p = this.property || "text";
36447     var sortType = this.sortType;
36448     var fs = this.folderSort;
36449     var cs = this.caseSensitive === true;
36450     var leafAttr = this.leafAttr || 'leaf';
36451
36452     this.sortFn = function(n1, n2){
36453         if(fs){
36454             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36455                 return 1;
36456             }
36457             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36458                 return -1;
36459             }
36460         }
36461         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36462         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36463         if(v1 < v2){
36464                         return dsc ? +1 : -1;
36465                 }else if(v1 > v2){
36466                         return dsc ? -1 : +1;
36467         }else{
36468                 return 0;
36469         }
36470     };
36471 };
36472
36473 Roo.tree.TreeSorter.prototype = {
36474     doSort : function(node){
36475         node.sort(this.sortFn);
36476     },
36477     
36478     compareNodes : function(n1, n2){
36479         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36480     },
36481     
36482     updateSort : function(tree, node){
36483         if(node.childrenRendered){
36484             this.doSort.defer(1, this, [node]);
36485         }
36486     }
36487 };/*
36488  * Based on:
36489  * Ext JS Library 1.1.1
36490  * Copyright(c) 2006-2007, Ext JS, LLC.
36491  *
36492  * Originally Released Under LGPL - original licence link has changed is not relivant.
36493  *
36494  * Fork - LGPL
36495  * <script type="text/javascript">
36496  */
36497
36498 if(Roo.dd.DropZone){
36499     
36500 Roo.tree.TreeDropZone = function(tree, config){
36501     this.allowParentInsert = false;
36502     this.allowContainerDrop = false;
36503     this.appendOnly = false;
36504     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36505     this.tree = tree;
36506     this.lastInsertClass = "x-tree-no-status";
36507     this.dragOverData = {};
36508 };
36509
36510 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36511     ddGroup : "TreeDD",
36512     scroll:  true,
36513     
36514     expandDelay : 1000,
36515     
36516     expandNode : function(node){
36517         if(node.hasChildNodes() && !node.isExpanded()){
36518             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36519         }
36520     },
36521     
36522     queueExpand : function(node){
36523         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36524     },
36525     
36526     cancelExpand : function(){
36527         if(this.expandProcId){
36528             clearTimeout(this.expandProcId);
36529             this.expandProcId = false;
36530         }
36531     },
36532     
36533     isValidDropPoint : function(n, pt, dd, e, data){
36534         if(!n || !data){ return false; }
36535         var targetNode = n.node;
36536         var dropNode = data.node;
36537         // default drop rules
36538         if(!(targetNode && targetNode.isTarget && pt)){
36539             return false;
36540         }
36541         if(pt == "append" && targetNode.allowChildren === false){
36542             return false;
36543         }
36544         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36545             return false;
36546         }
36547         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36548             return false;
36549         }
36550         // reuse the object
36551         var overEvent = this.dragOverData;
36552         overEvent.tree = this.tree;
36553         overEvent.target = targetNode;
36554         overEvent.data = data;
36555         overEvent.point = pt;
36556         overEvent.source = dd;
36557         overEvent.rawEvent = e;
36558         overEvent.dropNode = dropNode;
36559         overEvent.cancel = false;  
36560         var result = this.tree.fireEvent("nodedragover", overEvent);
36561         return overEvent.cancel === false && result !== false;
36562     },
36563     
36564     getDropPoint : function(e, n, dd)
36565     {
36566         var tn = n.node;
36567         if(tn.isRoot){
36568             return tn.allowChildren !== false ? "append" : false; // always append for root
36569         }
36570         var dragEl = n.ddel;
36571         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36572         var y = Roo.lib.Event.getPageY(e);
36573         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36574         
36575         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36576         var noAppend = tn.allowChildren === false;
36577         if(this.appendOnly || tn.parentNode.allowChildren === false){
36578             return noAppend ? false : "append";
36579         }
36580         var noBelow = false;
36581         if(!this.allowParentInsert){
36582             noBelow = tn.hasChildNodes() && tn.isExpanded();
36583         }
36584         var q = (b - t) / (noAppend ? 2 : 3);
36585         if(y >= t && y < (t + q)){
36586             return "above";
36587         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36588             return "below";
36589         }else{
36590             return "append";
36591         }
36592     },
36593     
36594     onNodeEnter : function(n, dd, e, data)
36595     {
36596         this.cancelExpand();
36597     },
36598     
36599     onNodeOver : function(n, dd, e, data)
36600     {
36601        
36602         var pt = this.getDropPoint(e, n, dd);
36603         var node = n.node;
36604         
36605         // auto node expand check
36606         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36607             this.queueExpand(node);
36608         }else if(pt != "append"){
36609             this.cancelExpand();
36610         }
36611         
36612         // set the insert point style on the target node
36613         var returnCls = this.dropNotAllowed;
36614         if(this.isValidDropPoint(n, pt, dd, e, data)){
36615            if(pt){
36616                var el = n.ddel;
36617                var cls;
36618                if(pt == "above"){
36619                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36620                    cls = "x-tree-drag-insert-above";
36621                }else if(pt == "below"){
36622                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36623                    cls = "x-tree-drag-insert-below";
36624                }else{
36625                    returnCls = "x-tree-drop-ok-append";
36626                    cls = "x-tree-drag-append";
36627                }
36628                if(this.lastInsertClass != cls){
36629                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36630                    this.lastInsertClass = cls;
36631                }
36632            }
36633        }
36634        return returnCls;
36635     },
36636     
36637     onNodeOut : function(n, dd, e, data){
36638         
36639         this.cancelExpand();
36640         this.removeDropIndicators(n);
36641     },
36642     
36643     onNodeDrop : function(n, dd, e, data){
36644         var point = this.getDropPoint(e, n, dd);
36645         var targetNode = n.node;
36646         targetNode.ui.startDrop();
36647         if(!this.isValidDropPoint(n, point, dd, e, data)){
36648             targetNode.ui.endDrop();
36649             return false;
36650         }
36651         // first try to find the drop node
36652         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36653         var dropEvent = {
36654             tree : this.tree,
36655             target: targetNode,
36656             data: data,
36657             point: point,
36658             source: dd,
36659             rawEvent: e,
36660             dropNode: dropNode,
36661             cancel: !dropNode   
36662         };
36663         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36664         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36665             targetNode.ui.endDrop();
36666             return false;
36667         }
36668         // allow target changing
36669         targetNode = dropEvent.target;
36670         if(point == "append" && !targetNode.isExpanded()){
36671             targetNode.expand(false, null, function(){
36672                 this.completeDrop(dropEvent);
36673             }.createDelegate(this));
36674         }else{
36675             this.completeDrop(dropEvent);
36676         }
36677         return true;
36678     },
36679     
36680     completeDrop : function(de){
36681         var ns = de.dropNode, p = de.point, t = de.target;
36682         if(!(ns instanceof Array)){
36683             ns = [ns];
36684         }
36685         var n;
36686         for(var i = 0, len = ns.length; i < len; i++){
36687             n = ns[i];
36688             if(p == "above"){
36689                 t.parentNode.insertBefore(n, t);
36690             }else if(p == "below"){
36691                 t.parentNode.insertBefore(n, t.nextSibling);
36692             }else{
36693                 t.appendChild(n);
36694             }
36695         }
36696         n.ui.focus();
36697         if(this.tree.hlDrop){
36698             n.ui.highlight();
36699         }
36700         t.ui.endDrop();
36701         this.tree.fireEvent("nodedrop", de);
36702     },
36703     
36704     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36705         if(this.tree.hlDrop){
36706             dropNode.ui.focus();
36707             dropNode.ui.highlight();
36708         }
36709         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36710     },
36711     
36712     getTree : function(){
36713         return this.tree;
36714     },
36715     
36716     removeDropIndicators : function(n){
36717         if(n && n.ddel){
36718             var el = n.ddel;
36719             Roo.fly(el).removeClass([
36720                     "x-tree-drag-insert-above",
36721                     "x-tree-drag-insert-below",
36722                     "x-tree-drag-append"]);
36723             this.lastInsertClass = "_noclass";
36724         }
36725     },
36726     
36727     beforeDragDrop : function(target, e, id){
36728         this.cancelExpand();
36729         return true;
36730     },
36731     
36732     afterRepair : function(data){
36733         if(data && Roo.enableFx){
36734             data.node.ui.highlight();
36735         }
36736         this.hideProxy();
36737     } 
36738     
36739 });
36740
36741 }
36742 /*
36743  * Based on:
36744  * Ext JS Library 1.1.1
36745  * Copyright(c) 2006-2007, Ext JS, LLC.
36746  *
36747  * Originally Released Under LGPL - original licence link has changed is not relivant.
36748  *
36749  * Fork - LGPL
36750  * <script type="text/javascript">
36751  */
36752  
36753
36754 if(Roo.dd.DragZone){
36755 Roo.tree.TreeDragZone = function(tree, config){
36756     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36757     this.tree = tree;
36758 };
36759
36760 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36761     ddGroup : "TreeDD",
36762    
36763     onBeforeDrag : function(data, e){
36764         var n = data.node;
36765         return n && n.draggable && !n.disabled;
36766     },
36767      
36768     
36769     onInitDrag : function(e){
36770         var data = this.dragData;
36771         this.tree.getSelectionModel().select(data.node);
36772         this.proxy.update("");
36773         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36774         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36775     },
36776     
36777     getRepairXY : function(e, data){
36778         return data.node.ui.getDDRepairXY();
36779     },
36780     
36781     onEndDrag : function(data, e){
36782         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36783         
36784         
36785     },
36786     
36787     onValidDrop : function(dd, e, id){
36788         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36789         this.hideProxy();
36790     },
36791     
36792     beforeInvalidDrop : function(e, id){
36793         // this scrolls the original position back into view
36794         var sm = this.tree.getSelectionModel();
36795         sm.clearSelections();
36796         sm.select(this.dragData.node);
36797     }
36798 });
36799 }/*
36800  * Based on:
36801  * Ext JS Library 1.1.1
36802  * Copyright(c) 2006-2007, Ext JS, LLC.
36803  *
36804  * Originally Released Under LGPL - original licence link has changed is not relivant.
36805  *
36806  * Fork - LGPL
36807  * <script type="text/javascript">
36808  */
36809 /**
36810  * @class Roo.tree.TreeEditor
36811  * @extends Roo.Editor
36812  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36813  * as the editor field.
36814  * @constructor
36815  * @param {Object} config (used to be the tree panel.)
36816  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36817  * 
36818  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36819  * @cfg {Roo.form.TextField|Object} field The field configuration
36820  *
36821  * 
36822  */
36823 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36824     var tree = config;
36825     var field;
36826     if (oldconfig) { // old style..
36827         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36828     } else {
36829         // new style..
36830         tree = config.tree;
36831         config.field = config.field  || {};
36832         config.field.xtype = 'TextField';
36833         field = Roo.factory(config.field, Roo.form);
36834     }
36835     config = config || {};
36836     
36837     
36838     this.addEvents({
36839         /**
36840          * @event beforenodeedit
36841          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36842          * false from the handler of this event.
36843          * @param {Editor} this
36844          * @param {Roo.tree.Node} node 
36845          */
36846         "beforenodeedit" : true
36847     });
36848     
36849     //Roo.log(config);
36850     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36851
36852     this.tree = tree;
36853
36854     tree.on('beforeclick', this.beforeNodeClick, this);
36855     tree.getTreeEl().on('mousedown', this.hide, this);
36856     this.on('complete', this.updateNode, this);
36857     this.on('beforestartedit', this.fitToTree, this);
36858     this.on('startedit', this.bindScroll, this, {delay:10});
36859     this.on('specialkey', this.onSpecialKey, this);
36860 };
36861
36862 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36863     /**
36864      * @cfg {String} alignment
36865      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36866      */
36867     alignment: "l-l",
36868     // inherit
36869     autoSize: false,
36870     /**
36871      * @cfg {Boolean} hideEl
36872      * True to hide the bound element while the editor is displayed (defaults to false)
36873      */
36874     hideEl : false,
36875     /**
36876      * @cfg {String} cls
36877      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36878      */
36879     cls: "x-small-editor x-tree-editor",
36880     /**
36881      * @cfg {Boolean} shim
36882      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36883      */
36884     shim:false,
36885     // inherit
36886     shadow:"frame",
36887     /**
36888      * @cfg {Number} maxWidth
36889      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36890      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36891      * scroll and client offsets into account prior to each edit.
36892      */
36893     maxWidth: 250,
36894
36895     editDelay : 350,
36896
36897     // private
36898     fitToTree : function(ed, el){
36899         var td = this.tree.getTreeEl().dom, nd = el.dom;
36900         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36901             td.scrollLeft = nd.offsetLeft;
36902         }
36903         var w = Math.min(
36904                 this.maxWidth,
36905                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36906         this.setSize(w, '');
36907         
36908         return this.fireEvent('beforenodeedit', this, this.editNode);
36909         
36910     },
36911
36912     // private
36913     triggerEdit : function(node){
36914         this.completeEdit();
36915         this.editNode = node;
36916         this.startEdit(node.ui.textNode, node.text);
36917     },
36918
36919     // private
36920     bindScroll : function(){
36921         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36922     },
36923
36924     // private
36925     beforeNodeClick : function(node, e){
36926         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36927         this.lastClick = new Date();
36928         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36929             e.stopEvent();
36930             this.triggerEdit(node);
36931             return false;
36932         }
36933         return true;
36934     },
36935
36936     // private
36937     updateNode : function(ed, value){
36938         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36939         this.editNode.setText(value);
36940     },
36941
36942     // private
36943     onHide : function(){
36944         Roo.tree.TreeEditor.superclass.onHide.call(this);
36945         if(this.editNode){
36946             this.editNode.ui.focus();
36947         }
36948     },
36949
36950     // private
36951     onSpecialKey : function(field, e){
36952         var k = e.getKey();
36953         if(k == e.ESC){
36954             e.stopEvent();
36955             this.cancelEdit();
36956         }else if(k == e.ENTER && !e.hasModifier()){
36957             e.stopEvent();
36958             this.completeEdit();
36959         }
36960     }
36961 });//<Script type="text/javascript">
36962 /*
36963  * Based on:
36964  * Ext JS Library 1.1.1
36965  * Copyright(c) 2006-2007, Ext JS, LLC.
36966  *
36967  * Originally Released Under LGPL - original licence link has changed is not relivant.
36968  *
36969  * Fork - LGPL
36970  * <script type="text/javascript">
36971  */
36972  
36973 /**
36974  * Not documented??? - probably should be...
36975  */
36976
36977 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
36978     //focus: Roo.emptyFn, // prevent odd scrolling behavior
36979     
36980     renderElements : function(n, a, targetNode, bulkRender){
36981         //consel.log("renderElements?");
36982         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
36983
36984         var t = n.getOwnerTree();
36985         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
36986         
36987         var cols = t.columns;
36988         var bw = t.borderWidth;
36989         var c = cols[0];
36990         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
36991          var cb = typeof a.checked == "boolean";
36992         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
36993         var colcls = 'x-t-' + tid + '-c0';
36994         var buf = [
36995             '<li class="x-tree-node">',
36996             
36997                 
36998                 '<div class="x-tree-node-el ', a.cls,'">',
36999                     // extran...
37000                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37001                 
37002                 
37003                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37004                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37005                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37006                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37007                            (a.iconCls ? ' '+a.iconCls : ''),
37008                            '" unselectable="on" />',
37009                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37010                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37011                              
37012                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37013                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37014                             '<span unselectable="on" qtip="' + tx + '">',
37015                              tx,
37016                              '</span></a>' ,
37017                     '</div>',
37018                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37019                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37020                  ];
37021         for(var i = 1, len = cols.length; i < len; i++){
37022             c = cols[i];
37023             colcls = 'x-t-' + tid + '-c' +i;
37024             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37025             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37026                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37027                       "</div>");
37028          }
37029          
37030          buf.push(
37031             '</a>',
37032             '<div class="x-clear"></div></div>',
37033             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37034             "</li>");
37035         
37036         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37037             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37038                                 n.nextSibling.ui.getEl(), buf.join(""));
37039         }else{
37040             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37041         }
37042         var el = this.wrap.firstChild;
37043         this.elRow = el;
37044         this.elNode = el.firstChild;
37045         this.ranchor = el.childNodes[1];
37046         this.ctNode = this.wrap.childNodes[1];
37047         var cs = el.firstChild.childNodes;
37048         this.indentNode = cs[0];
37049         this.ecNode = cs[1];
37050         this.iconNode = cs[2];
37051         var index = 3;
37052         if(cb){
37053             this.checkbox = cs[3];
37054             index++;
37055         }
37056         this.anchor = cs[index];
37057         
37058         this.textNode = cs[index].firstChild;
37059         
37060         //el.on("click", this.onClick, this);
37061         //el.on("dblclick", this.onDblClick, this);
37062         
37063         
37064        // console.log(this);
37065     },
37066     initEvents : function(){
37067         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37068         
37069             
37070         var a = this.ranchor;
37071
37072         var el = Roo.get(a);
37073
37074         if(Roo.isOpera){ // opera render bug ignores the CSS
37075             el.setStyle("text-decoration", "none");
37076         }
37077
37078         el.on("click", this.onClick, this);
37079         el.on("dblclick", this.onDblClick, this);
37080         el.on("contextmenu", this.onContextMenu, this);
37081         
37082     },
37083     
37084     /*onSelectedChange : function(state){
37085         if(state){
37086             this.focus();
37087             this.addClass("x-tree-selected");
37088         }else{
37089             //this.blur();
37090             this.removeClass("x-tree-selected");
37091         }
37092     },*/
37093     addClass : function(cls){
37094         if(this.elRow){
37095             Roo.fly(this.elRow).addClass(cls);
37096         }
37097         
37098     },
37099     
37100     
37101     removeClass : function(cls){
37102         if(this.elRow){
37103             Roo.fly(this.elRow).removeClass(cls);
37104         }
37105     }
37106
37107     
37108     
37109 });//<Script type="text/javascript">
37110
37111 /*
37112  * Based on:
37113  * Ext JS Library 1.1.1
37114  * Copyright(c) 2006-2007, Ext JS, LLC.
37115  *
37116  * Originally Released Under LGPL - original licence link has changed is not relivant.
37117  *
37118  * Fork - LGPL
37119  * <script type="text/javascript">
37120  */
37121  
37122
37123 /**
37124  * @class Roo.tree.ColumnTree
37125  * @extends Roo.data.TreePanel
37126  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37127  * @cfg {int} borderWidth  compined right/left border allowance
37128  * @constructor
37129  * @param {String/HTMLElement/Element} el The container element
37130  * @param {Object} config
37131  */
37132 Roo.tree.ColumnTree =  function(el, config)
37133 {
37134    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37135    this.addEvents({
37136         /**
37137         * @event resize
37138         * Fire this event on a container when it resizes
37139         * @param {int} w Width
37140         * @param {int} h Height
37141         */
37142        "resize" : true
37143     });
37144     this.on('resize', this.onResize, this);
37145 };
37146
37147 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37148     //lines:false,
37149     
37150     
37151     borderWidth: Roo.isBorderBox ? 0 : 2, 
37152     headEls : false,
37153     
37154     render : function(){
37155         // add the header.....
37156        
37157         Roo.tree.ColumnTree.superclass.render.apply(this);
37158         
37159         this.el.addClass('x-column-tree');
37160         
37161         this.headers = this.el.createChild(
37162             {cls:'x-tree-headers'},this.innerCt.dom);
37163    
37164         var cols = this.columns, c;
37165         var totalWidth = 0;
37166         this.headEls = [];
37167         var  len = cols.length;
37168         for(var i = 0; i < len; i++){
37169              c = cols[i];
37170              totalWidth += c.width;
37171             this.headEls.push(this.headers.createChild({
37172                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37173                  cn: {
37174                      cls:'x-tree-hd-text',
37175                      html: c.header
37176                  },
37177                  style:'width:'+(c.width-this.borderWidth)+'px;'
37178              }));
37179         }
37180         this.headers.createChild({cls:'x-clear'});
37181         // prevent floats from wrapping when clipped
37182         this.headers.setWidth(totalWidth);
37183         //this.innerCt.setWidth(totalWidth);
37184         this.innerCt.setStyle({ overflow: 'auto' });
37185         this.onResize(this.width, this.height);
37186              
37187         
37188     },
37189     onResize : function(w,h)
37190     {
37191         this.height = h;
37192         this.width = w;
37193         // resize cols..
37194         this.innerCt.setWidth(this.width);
37195         this.innerCt.setHeight(this.height-20);
37196         
37197         // headers...
37198         var cols = this.columns, c;
37199         var totalWidth = 0;
37200         var expEl = false;
37201         var len = cols.length;
37202         for(var i = 0; i < len; i++){
37203             c = cols[i];
37204             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37205                 // it's the expander..
37206                 expEl  = this.headEls[i];
37207                 continue;
37208             }
37209             totalWidth += c.width;
37210             
37211         }
37212         if (expEl) {
37213             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37214         }
37215         this.headers.setWidth(w-20);
37216
37217         
37218         
37219         
37220     }
37221 });
37222 /*
37223  * Based on:
37224  * Ext JS Library 1.1.1
37225  * Copyright(c) 2006-2007, Ext JS, LLC.
37226  *
37227  * Originally Released Under LGPL - original licence link has changed is not relivant.
37228  *
37229  * Fork - LGPL
37230  * <script type="text/javascript">
37231  */
37232  
37233 /**
37234  * @class Roo.menu.Menu
37235  * @extends Roo.util.Observable
37236  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37237  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37238  * @constructor
37239  * Creates a new Menu
37240  * @param {Object} config Configuration options
37241  */
37242 Roo.menu.Menu = function(config){
37243     
37244     Roo.menu.Menu.superclass.constructor.call(this, config);
37245     
37246     this.id = this.id || Roo.id();
37247     this.addEvents({
37248         /**
37249          * @event beforeshow
37250          * Fires before this menu is displayed
37251          * @param {Roo.menu.Menu} this
37252          */
37253         beforeshow : true,
37254         /**
37255          * @event beforehide
37256          * Fires before this menu is hidden
37257          * @param {Roo.menu.Menu} this
37258          */
37259         beforehide : true,
37260         /**
37261          * @event show
37262          * Fires after this menu is displayed
37263          * @param {Roo.menu.Menu} this
37264          */
37265         show : true,
37266         /**
37267          * @event hide
37268          * Fires after this menu is hidden
37269          * @param {Roo.menu.Menu} this
37270          */
37271         hide : true,
37272         /**
37273          * @event click
37274          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37275          * @param {Roo.menu.Menu} this
37276          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37277          * @param {Roo.EventObject} e
37278          */
37279         click : true,
37280         /**
37281          * @event mouseover
37282          * Fires when the mouse is hovering over this menu
37283          * @param {Roo.menu.Menu} this
37284          * @param {Roo.EventObject} e
37285          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37286          */
37287         mouseover : true,
37288         /**
37289          * @event mouseout
37290          * Fires when the mouse exits this menu
37291          * @param {Roo.menu.Menu} this
37292          * @param {Roo.EventObject} e
37293          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37294          */
37295         mouseout : true,
37296         /**
37297          * @event itemclick
37298          * Fires when a menu item contained in this menu is clicked
37299          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37300          * @param {Roo.EventObject} e
37301          */
37302         itemclick: true
37303     });
37304     if (this.registerMenu) {
37305         Roo.menu.MenuMgr.register(this);
37306     }
37307     
37308     var mis = this.items;
37309     this.items = new Roo.util.MixedCollection();
37310     if(mis){
37311         this.add.apply(this, mis);
37312     }
37313 };
37314
37315 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37316     /**
37317      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37318      */
37319     minWidth : 120,
37320     /**
37321      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37322      * for bottom-right shadow (defaults to "sides")
37323      */
37324     shadow : "sides",
37325     /**
37326      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37327      * this menu (defaults to "tl-tr?")
37328      */
37329     subMenuAlign : "tl-tr?",
37330     /**
37331      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37332      * relative to its element of origin (defaults to "tl-bl?")
37333      */
37334     defaultAlign : "tl-bl?",
37335     /**
37336      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37337      */
37338     allowOtherMenus : false,
37339     /**
37340      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37341      */
37342     registerMenu : true,
37343
37344     hidden:true,
37345
37346     // private
37347     render : function(){
37348         if(this.el){
37349             return;
37350         }
37351         var el = this.el = new Roo.Layer({
37352             cls: "x-menu",
37353             shadow:this.shadow,
37354             constrain: false,
37355             parentEl: this.parentEl || document.body,
37356             zindex:15000
37357         });
37358
37359         this.keyNav = new Roo.menu.MenuNav(this);
37360
37361         if(this.plain){
37362             el.addClass("x-menu-plain");
37363         }
37364         if(this.cls){
37365             el.addClass(this.cls);
37366         }
37367         // generic focus element
37368         this.focusEl = el.createChild({
37369             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37370         });
37371         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37372         //disabling touch- as it's causing issues ..
37373         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37374         ul.on('click'   , this.onClick, this);
37375         
37376         
37377         ul.on("mouseover", this.onMouseOver, this);
37378         ul.on("mouseout", this.onMouseOut, this);
37379         this.items.each(function(item){
37380             if (item.hidden) {
37381                 return;
37382             }
37383             
37384             var li = document.createElement("li");
37385             li.className = "x-menu-list-item";
37386             ul.dom.appendChild(li);
37387             item.render(li, this);
37388         }, this);
37389         this.ul = ul;
37390         this.autoWidth();
37391     },
37392
37393     // private
37394     autoWidth : function(){
37395         var el = this.el, ul = this.ul;
37396         if(!el){
37397             return;
37398         }
37399         var w = this.width;
37400         if(w){
37401             el.setWidth(w);
37402         }else if(Roo.isIE){
37403             el.setWidth(this.minWidth);
37404             var t = el.dom.offsetWidth; // force recalc
37405             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37406         }
37407     },
37408
37409     // private
37410     delayAutoWidth : function(){
37411         if(this.rendered){
37412             if(!this.awTask){
37413                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37414             }
37415             this.awTask.delay(20);
37416         }
37417     },
37418
37419     // private
37420     findTargetItem : function(e){
37421         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37422         if(t && t.menuItemId){
37423             return this.items.get(t.menuItemId);
37424         }
37425     },
37426
37427     // private
37428     onClick : function(e){
37429         Roo.log("menu.onClick");
37430         var t = this.findTargetItem(e);
37431         if(!t){
37432             return;
37433         }
37434         Roo.log(e);
37435         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37436             if(t == this.activeItem && t.shouldDeactivate(e)){
37437                 this.activeItem.deactivate();
37438                 delete this.activeItem;
37439                 return;
37440             }
37441             if(t.canActivate){
37442                 this.setActiveItem(t, true);
37443             }
37444             return;
37445             
37446             
37447         }
37448         
37449         t.onClick(e);
37450         this.fireEvent("click", this, t, e);
37451     },
37452
37453     // private
37454     setActiveItem : function(item, autoExpand){
37455         if(item != this.activeItem){
37456             if(this.activeItem){
37457                 this.activeItem.deactivate();
37458             }
37459             this.activeItem = item;
37460             item.activate(autoExpand);
37461         }else if(autoExpand){
37462             item.expandMenu();
37463         }
37464     },
37465
37466     // private
37467     tryActivate : function(start, step){
37468         var items = this.items;
37469         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37470             var item = items.get(i);
37471             if(!item.disabled && item.canActivate){
37472                 this.setActiveItem(item, false);
37473                 return item;
37474             }
37475         }
37476         return false;
37477     },
37478
37479     // private
37480     onMouseOver : function(e){
37481         var t;
37482         if(t = this.findTargetItem(e)){
37483             if(t.canActivate && !t.disabled){
37484                 this.setActiveItem(t, true);
37485             }
37486         }
37487         this.fireEvent("mouseover", this, e, t);
37488     },
37489
37490     // private
37491     onMouseOut : function(e){
37492         var t;
37493         if(t = this.findTargetItem(e)){
37494             if(t == this.activeItem && t.shouldDeactivate(e)){
37495                 this.activeItem.deactivate();
37496                 delete this.activeItem;
37497             }
37498         }
37499         this.fireEvent("mouseout", this, e, t);
37500     },
37501
37502     /**
37503      * Read-only.  Returns true if the menu is currently displayed, else false.
37504      * @type Boolean
37505      */
37506     isVisible : function(){
37507         return this.el && !this.hidden;
37508     },
37509
37510     /**
37511      * Displays this menu relative to another element
37512      * @param {String/HTMLElement/Roo.Element} element The element to align to
37513      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37514      * the element (defaults to this.defaultAlign)
37515      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37516      */
37517     show : function(el, pos, parentMenu){
37518         this.parentMenu = parentMenu;
37519         if(!this.el){
37520             this.render();
37521         }
37522         this.fireEvent("beforeshow", this);
37523         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37524     },
37525
37526     /**
37527      * Displays this menu at a specific xy position
37528      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37529      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37530      */
37531     showAt : function(xy, parentMenu, /* private: */_e){
37532         this.parentMenu = parentMenu;
37533         if(!this.el){
37534             this.render();
37535         }
37536         if(_e !== false){
37537             this.fireEvent("beforeshow", this);
37538             xy = this.el.adjustForConstraints(xy);
37539         }
37540         this.el.setXY(xy);
37541         this.el.show();
37542         this.hidden = false;
37543         this.focus();
37544         this.fireEvent("show", this);
37545     },
37546
37547     focus : function(){
37548         if(!this.hidden){
37549             this.doFocus.defer(50, this);
37550         }
37551     },
37552
37553     doFocus : function(){
37554         if(!this.hidden){
37555             this.focusEl.focus();
37556         }
37557     },
37558
37559     /**
37560      * Hides this menu and optionally all parent menus
37561      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37562      */
37563     hide : function(deep){
37564         if(this.el && this.isVisible()){
37565             this.fireEvent("beforehide", this);
37566             if(this.activeItem){
37567                 this.activeItem.deactivate();
37568                 this.activeItem = null;
37569             }
37570             this.el.hide();
37571             this.hidden = true;
37572             this.fireEvent("hide", this);
37573         }
37574         if(deep === true && this.parentMenu){
37575             this.parentMenu.hide(true);
37576         }
37577     },
37578
37579     /**
37580      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37581      * Any of the following are valid:
37582      * <ul>
37583      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37584      * <li>An HTMLElement object which will be converted to a menu item</li>
37585      * <li>A menu item config object that will be created as a new menu item</li>
37586      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37587      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37588      * </ul>
37589      * Usage:
37590      * <pre><code>
37591 // Create the menu
37592 var menu = new Roo.menu.Menu();
37593
37594 // Create a menu item to add by reference
37595 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37596
37597 // Add a bunch of items at once using different methods.
37598 // Only the last item added will be returned.
37599 var item = menu.add(
37600     menuItem,                // add existing item by ref
37601     'Dynamic Item',          // new TextItem
37602     '-',                     // new separator
37603     { text: 'Config Item' }  // new item by config
37604 );
37605 </code></pre>
37606      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37607      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37608      */
37609     add : function(){
37610         var a = arguments, l = a.length, item;
37611         for(var i = 0; i < l; i++){
37612             var el = a[i];
37613             if ((typeof(el) == "object") && el.xtype && el.xns) {
37614                 el = Roo.factory(el, Roo.menu);
37615             }
37616             
37617             if(el.render){ // some kind of Item
37618                 item = this.addItem(el);
37619             }else if(typeof el == "string"){ // string
37620                 if(el == "separator" || el == "-"){
37621                     item = this.addSeparator();
37622                 }else{
37623                     item = this.addText(el);
37624                 }
37625             }else if(el.tagName || el.el){ // element
37626                 item = this.addElement(el);
37627             }else if(typeof el == "object"){ // must be menu item config?
37628                 item = this.addMenuItem(el);
37629             }
37630         }
37631         return item;
37632     },
37633
37634     /**
37635      * Returns this menu's underlying {@link Roo.Element} object
37636      * @return {Roo.Element} The element
37637      */
37638     getEl : function(){
37639         if(!this.el){
37640             this.render();
37641         }
37642         return this.el;
37643     },
37644
37645     /**
37646      * Adds a separator bar to the menu
37647      * @return {Roo.menu.Item} The menu item that was added
37648      */
37649     addSeparator : function(){
37650         return this.addItem(new Roo.menu.Separator());
37651     },
37652
37653     /**
37654      * Adds an {@link Roo.Element} object to the menu
37655      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37656      * @return {Roo.menu.Item} The menu item that was added
37657      */
37658     addElement : function(el){
37659         return this.addItem(new Roo.menu.BaseItem(el));
37660     },
37661
37662     /**
37663      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37664      * @param {Roo.menu.Item} item The menu item to add
37665      * @return {Roo.menu.Item} The menu item that was added
37666      */
37667     addItem : function(item){
37668         this.items.add(item);
37669         if(this.ul){
37670             var li = document.createElement("li");
37671             li.className = "x-menu-list-item";
37672             this.ul.dom.appendChild(li);
37673             item.render(li, this);
37674             this.delayAutoWidth();
37675         }
37676         return item;
37677     },
37678
37679     /**
37680      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37681      * @param {Object} config A MenuItem config object
37682      * @return {Roo.menu.Item} The menu item that was added
37683      */
37684     addMenuItem : function(config){
37685         if(!(config instanceof Roo.menu.Item)){
37686             if(typeof config.checked == "boolean"){ // must be check menu item config?
37687                 config = new Roo.menu.CheckItem(config);
37688             }else{
37689                 config = new Roo.menu.Item(config);
37690             }
37691         }
37692         return this.addItem(config);
37693     },
37694
37695     /**
37696      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37697      * @param {String} text The text to display in the menu item
37698      * @return {Roo.menu.Item} The menu item that was added
37699      */
37700     addText : function(text){
37701         return this.addItem(new Roo.menu.TextItem({ text : text }));
37702     },
37703
37704     /**
37705      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37706      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37707      * @param {Roo.menu.Item} item The menu item to add
37708      * @return {Roo.menu.Item} The menu item that was added
37709      */
37710     insert : function(index, item){
37711         this.items.insert(index, item);
37712         if(this.ul){
37713             var li = document.createElement("li");
37714             li.className = "x-menu-list-item";
37715             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37716             item.render(li, this);
37717             this.delayAutoWidth();
37718         }
37719         return item;
37720     },
37721
37722     /**
37723      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37724      * @param {Roo.menu.Item} item The menu item to remove
37725      */
37726     remove : function(item){
37727         this.items.removeKey(item.id);
37728         item.destroy();
37729     },
37730
37731     /**
37732      * Removes and destroys all items in the menu
37733      */
37734     removeAll : function(){
37735         var f;
37736         while(f = this.items.first()){
37737             this.remove(f);
37738         }
37739     }
37740 });
37741
37742 // MenuNav is a private utility class used internally by the Menu
37743 Roo.menu.MenuNav = function(menu){
37744     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37745     this.scope = this.menu = menu;
37746 };
37747
37748 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37749     doRelay : function(e, h){
37750         var k = e.getKey();
37751         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37752             this.menu.tryActivate(0, 1);
37753             return false;
37754         }
37755         return h.call(this.scope || this, e, this.menu);
37756     },
37757
37758     up : function(e, m){
37759         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37760             m.tryActivate(m.items.length-1, -1);
37761         }
37762     },
37763
37764     down : function(e, m){
37765         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37766             m.tryActivate(0, 1);
37767         }
37768     },
37769
37770     right : function(e, m){
37771         if(m.activeItem){
37772             m.activeItem.expandMenu(true);
37773         }
37774     },
37775
37776     left : function(e, m){
37777         m.hide();
37778         if(m.parentMenu && m.parentMenu.activeItem){
37779             m.parentMenu.activeItem.activate();
37780         }
37781     },
37782
37783     enter : function(e, m){
37784         if(m.activeItem){
37785             e.stopPropagation();
37786             m.activeItem.onClick(e);
37787             m.fireEvent("click", this, m.activeItem);
37788             return true;
37789         }
37790     }
37791 });/*
37792  * Based on:
37793  * Ext JS Library 1.1.1
37794  * Copyright(c) 2006-2007, Ext JS, LLC.
37795  *
37796  * Originally Released Under LGPL - original licence link has changed is not relivant.
37797  *
37798  * Fork - LGPL
37799  * <script type="text/javascript">
37800  */
37801  
37802 /**
37803  * @class Roo.menu.MenuMgr
37804  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37805  * @singleton
37806  */
37807 Roo.menu.MenuMgr = function(){
37808    var menus, active, groups = {}, attached = false, lastShow = new Date();
37809
37810    // private - called when first menu is created
37811    function init(){
37812        menus = {};
37813        active = new Roo.util.MixedCollection();
37814        Roo.get(document).addKeyListener(27, function(){
37815            if(active.length > 0){
37816                hideAll();
37817            }
37818        });
37819    }
37820
37821    // private
37822    function hideAll(){
37823        if(active && active.length > 0){
37824            var c = active.clone();
37825            c.each(function(m){
37826                m.hide();
37827            });
37828        }
37829    }
37830
37831    // private
37832    function onHide(m){
37833        active.remove(m);
37834        if(active.length < 1){
37835            Roo.get(document).un("mousedown", onMouseDown);
37836            attached = false;
37837        }
37838    }
37839
37840    // private
37841    function onShow(m){
37842        var last = active.last();
37843        lastShow = new Date();
37844        active.add(m);
37845        if(!attached){
37846            Roo.get(document).on("mousedown", onMouseDown);
37847            attached = true;
37848        }
37849        if(m.parentMenu){
37850           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37851           m.parentMenu.activeChild = m;
37852        }else if(last && last.isVisible()){
37853           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37854        }
37855    }
37856
37857    // private
37858    function onBeforeHide(m){
37859        if(m.activeChild){
37860            m.activeChild.hide();
37861        }
37862        if(m.autoHideTimer){
37863            clearTimeout(m.autoHideTimer);
37864            delete m.autoHideTimer;
37865        }
37866    }
37867
37868    // private
37869    function onBeforeShow(m){
37870        var pm = m.parentMenu;
37871        if(!pm && !m.allowOtherMenus){
37872            hideAll();
37873        }else if(pm && pm.activeChild && active != m){
37874            pm.activeChild.hide();
37875        }
37876    }
37877
37878    // private
37879    function onMouseDown(e){
37880        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37881            hideAll();
37882        }
37883    }
37884
37885    // private
37886    function onBeforeCheck(mi, state){
37887        if(state){
37888            var g = groups[mi.group];
37889            for(var i = 0, l = g.length; i < l; i++){
37890                if(g[i] != mi){
37891                    g[i].setChecked(false);
37892                }
37893            }
37894        }
37895    }
37896
37897    return {
37898
37899        /**
37900         * Hides all menus that are currently visible
37901         */
37902        hideAll : function(){
37903             hideAll();  
37904        },
37905
37906        // private
37907        register : function(menu){
37908            if(!menus){
37909                init();
37910            }
37911            menus[menu.id] = menu;
37912            menu.on("beforehide", onBeforeHide);
37913            menu.on("hide", onHide);
37914            menu.on("beforeshow", onBeforeShow);
37915            menu.on("show", onShow);
37916            var g = menu.group;
37917            if(g && menu.events["checkchange"]){
37918                if(!groups[g]){
37919                    groups[g] = [];
37920                }
37921                groups[g].push(menu);
37922                menu.on("checkchange", onCheck);
37923            }
37924        },
37925
37926         /**
37927          * Returns a {@link Roo.menu.Menu} object
37928          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37929          * be used to generate and return a new Menu instance.
37930          */
37931        get : function(menu){
37932            if(typeof menu == "string"){ // menu id
37933                return menus[menu];
37934            }else if(menu.events){  // menu instance
37935                return menu;
37936            }else if(typeof menu.length == 'number'){ // array of menu items?
37937                return new Roo.menu.Menu({items:menu});
37938            }else{ // otherwise, must be a config
37939                return new Roo.menu.Menu(menu);
37940            }
37941        },
37942
37943        // private
37944        unregister : function(menu){
37945            delete menus[menu.id];
37946            menu.un("beforehide", onBeforeHide);
37947            menu.un("hide", onHide);
37948            menu.un("beforeshow", onBeforeShow);
37949            menu.un("show", onShow);
37950            var g = menu.group;
37951            if(g && menu.events["checkchange"]){
37952                groups[g].remove(menu);
37953                menu.un("checkchange", onCheck);
37954            }
37955        },
37956
37957        // private
37958        registerCheckable : function(menuItem){
37959            var g = menuItem.group;
37960            if(g){
37961                if(!groups[g]){
37962                    groups[g] = [];
37963                }
37964                groups[g].push(menuItem);
37965                menuItem.on("beforecheckchange", onBeforeCheck);
37966            }
37967        },
37968
37969        // private
37970        unregisterCheckable : function(menuItem){
37971            var g = menuItem.group;
37972            if(g){
37973                groups[g].remove(menuItem);
37974                menuItem.un("beforecheckchange", onBeforeCheck);
37975            }
37976        }
37977    };
37978 }();/*
37979  * Based on:
37980  * Ext JS Library 1.1.1
37981  * Copyright(c) 2006-2007, Ext JS, LLC.
37982  *
37983  * Originally Released Under LGPL - original licence link has changed is not relivant.
37984  *
37985  * Fork - LGPL
37986  * <script type="text/javascript">
37987  */
37988  
37989
37990 /**
37991  * @class Roo.menu.BaseItem
37992  * @extends Roo.Component
37993  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
37994  * management and base configuration options shared by all menu components.
37995  * @constructor
37996  * Creates a new BaseItem
37997  * @param {Object} config Configuration options
37998  */
37999 Roo.menu.BaseItem = function(config){
38000     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38001
38002     this.addEvents({
38003         /**
38004          * @event click
38005          * Fires when this item is clicked
38006          * @param {Roo.menu.BaseItem} this
38007          * @param {Roo.EventObject} e
38008          */
38009         click: true,
38010         /**
38011          * @event activate
38012          * Fires when this item is activated
38013          * @param {Roo.menu.BaseItem} this
38014          */
38015         activate : true,
38016         /**
38017          * @event deactivate
38018          * Fires when this item is deactivated
38019          * @param {Roo.menu.BaseItem} this
38020          */
38021         deactivate : true
38022     });
38023
38024     if(this.handler){
38025         this.on("click", this.handler, this.scope, true);
38026     }
38027 };
38028
38029 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38030     /**
38031      * @cfg {Function} handler
38032      * A function that will handle the click event of this menu item (defaults to undefined)
38033      */
38034     /**
38035      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38036      */
38037     canActivate : false,
38038     
38039      /**
38040      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38041      */
38042     hidden: false,
38043     
38044     /**
38045      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38046      */
38047     activeClass : "x-menu-item-active",
38048     /**
38049      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38050      */
38051     hideOnClick : true,
38052     /**
38053      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38054      */
38055     hideDelay : 100,
38056
38057     // private
38058     ctype: "Roo.menu.BaseItem",
38059
38060     // private
38061     actionMode : "container",
38062
38063     // private
38064     render : function(container, parentMenu){
38065         this.parentMenu = parentMenu;
38066         Roo.menu.BaseItem.superclass.render.call(this, container);
38067         this.container.menuItemId = this.id;
38068     },
38069
38070     // private
38071     onRender : function(container, position){
38072         this.el = Roo.get(this.el);
38073         container.dom.appendChild(this.el.dom);
38074     },
38075
38076     // private
38077     onClick : function(e){
38078         if(!this.disabled && this.fireEvent("click", this, e) !== false
38079                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38080             this.handleClick(e);
38081         }else{
38082             e.stopEvent();
38083         }
38084     },
38085
38086     // private
38087     activate : function(){
38088         if(this.disabled){
38089             return false;
38090         }
38091         var li = this.container;
38092         li.addClass(this.activeClass);
38093         this.region = li.getRegion().adjust(2, 2, -2, -2);
38094         this.fireEvent("activate", this);
38095         return true;
38096     },
38097
38098     // private
38099     deactivate : function(){
38100         this.container.removeClass(this.activeClass);
38101         this.fireEvent("deactivate", this);
38102     },
38103
38104     // private
38105     shouldDeactivate : function(e){
38106         return !this.region || !this.region.contains(e.getPoint());
38107     },
38108
38109     // private
38110     handleClick : function(e){
38111         if(this.hideOnClick){
38112             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38113         }
38114     },
38115
38116     // private
38117     expandMenu : function(autoActivate){
38118         // do nothing
38119     },
38120
38121     // private
38122     hideMenu : function(){
38123         // do nothing
38124     }
38125 });/*
38126  * Based on:
38127  * Ext JS Library 1.1.1
38128  * Copyright(c) 2006-2007, Ext JS, LLC.
38129  *
38130  * Originally Released Under LGPL - original licence link has changed is not relivant.
38131  *
38132  * Fork - LGPL
38133  * <script type="text/javascript">
38134  */
38135  
38136 /**
38137  * @class Roo.menu.Adapter
38138  * @extends Roo.menu.BaseItem
38139  * 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.
38140  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38141  * @constructor
38142  * Creates a new Adapter
38143  * @param {Object} config Configuration options
38144  */
38145 Roo.menu.Adapter = function(component, config){
38146     Roo.menu.Adapter.superclass.constructor.call(this, config);
38147     this.component = component;
38148 };
38149 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38150     // private
38151     canActivate : true,
38152
38153     // private
38154     onRender : function(container, position){
38155         this.component.render(container);
38156         this.el = this.component.getEl();
38157     },
38158
38159     // private
38160     activate : function(){
38161         if(this.disabled){
38162             return false;
38163         }
38164         this.component.focus();
38165         this.fireEvent("activate", this);
38166         return true;
38167     },
38168
38169     // private
38170     deactivate : function(){
38171         this.fireEvent("deactivate", this);
38172     },
38173
38174     // private
38175     disable : function(){
38176         this.component.disable();
38177         Roo.menu.Adapter.superclass.disable.call(this);
38178     },
38179
38180     // private
38181     enable : function(){
38182         this.component.enable();
38183         Roo.menu.Adapter.superclass.enable.call(this);
38184     }
38185 });/*
38186  * Based on:
38187  * Ext JS Library 1.1.1
38188  * Copyright(c) 2006-2007, Ext JS, LLC.
38189  *
38190  * Originally Released Under LGPL - original licence link has changed is not relivant.
38191  *
38192  * Fork - LGPL
38193  * <script type="text/javascript">
38194  */
38195
38196 /**
38197  * @class Roo.menu.TextItem
38198  * @extends Roo.menu.BaseItem
38199  * Adds a static text string to a menu, usually used as either a heading or group separator.
38200  * Note: old style constructor with text is still supported.
38201  * 
38202  * @constructor
38203  * Creates a new TextItem
38204  * @param {Object} cfg Configuration
38205  */
38206 Roo.menu.TextItem = function(cfg){
38207     if (typeof(cfg) == 'string') {
38208         this.text = cfg;
38209     } else {
38210         Roo.apply(this,cfg);
38211     }
38212     
38213     Roo.menu.TextItem.superclass.constructor.call(this);
38214 };
38215
38216 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38217     /**
38218      * @cfg {Boolean} text Text to show on item.
38219      */
38220     text : '',
38221     
38222     /**
38223      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38224      */
38225     hideOnClick : false,
38226     /**
38227      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38228      */
38229     itemCls : "x-menu-text",
38230
38231     // private
38232     onRender : function(){
38233         var s = document.createElement("span");
38234         s.className = this.itemCls;
38235         s.innerHTML = this.text;
38236         this.el = s;
38237         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38238     }
38239 });/*
38240  * Based on:
38241  * Ext JS Library 1.1.1
38242  * Copyright(c) 2006-2007, Ext JS, LLC.
38243  *
38244  * Originally Released Under LGPL - original licence link has changed is not relivant.
38245  *
38246  * Fork - LGPL
38247  * <script type="text/javascript">
38248  */
38249
38250 /**
38251  * @class Roo.menu.Separator
38252  * @extends Roo.menu.BaseItem
38253  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38254  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38255  * @constructor
38256  * @param {Object} config Configuration options
38257  */
38258 Roo.menu.Separator = function(config){
38259     Roo.menu.Separator.superclass.constructor.call(this, config);
38260 };
38261
38262 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38263     /**
38264      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38265      */
38266     itemCls : "x-menu-sep",
38267     /**
38268      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38269      */
38270     hideOnClick : false,
38271
38272     // private
38273     onRender : function(li){
38274         var s = document.createElement("span");
38275         s.className = this.itemCls;
38276         s.innerHTML = "&#160;";
38277         this.el = s;
38278         li.addClass("x-menu-sep-li");
38279         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38280     }
38281 });/*
38282  * Based on:
38283  * Ext JS Library 1.1.1
38284  * Copyright(c) 2006-2007, Ext JS, LLC.
38285  *
38286  * Originally Released Under LGPL - original licence link has changed is not relivant.
38287  *
38288  * Fork - LGPL
38289  * <script type="text/javascript">
38290  */
38291 /**
38292  * @class Roo.menu.Item
38293  * @extends Roo.menu.BaseItem
38294  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38295  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38296  * activation and click handling.
38297  * @constructor
38298  * Creates a new Item
38299  * @param {Object} config Configuration options
38300  */
38301 Roo.menu.Item = function(config){
38302     Roo.menu.Item.superclass.constructor.call(this, config);
38303     if(this.menu){
38304         this.menu = Roo.menu.MenuMgr.get(this.menu);
38305     }
38306 };
38307 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38308     
38309     /**
38310      * @cfg {String} text
38311      * The text to show on the menu item.
38312      */
38313     text: '',
38314      /**
38315      * @cfg {String} HTML to render in menu
38316      * The text to show on the menu item (HTML version).
38317      */
38318     html: '',
38319     /**
38320      * @cfg {String} icon
38321      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38322      */
38323     icon: undefined,
38324     /**
38325      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38326      */
38327     itemCls : "x-menu-item",
38328     /**
38329      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38330      */
38331     canActivate : true,
38332     /**
38333      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38334      */
38335     showDelay: 200,
38336     // doc'd in BaseItem
38337     hideDelay: 200,
38338
38339     // private
38340     ctype: "Roo.menu.Item",
38341     
38342     // private
38343     onRender : function(container, position){
38344         var el = document.createElement("a");
38345         el.hideFocus = true;
38346         el.unselectable = "on";
38347         el.href = this.href || "#";
38348         if(this.hrefTarget){
38349             el.target = this.hrefTarget;
38350         }
38351         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38352         
38353         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38354         
38355         el.innerHTML = String.format(
38356                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38357                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38358         this.el = el;
38359         Roo.menu.Item.superclass.onRender.call(this, container, position);
38360     },
38361
38362     /**
38363      * Sets the text to display in this menu item
38364      * @param {String} text The text to display
38365      * @param {Boolean} isHTML true to indicate text is pure html.
38366      */
38367     setText : function(text, isHTML){
38368         if (isHTML) {
38369             this.html = text;
38370         } else {
38371             this.text = text;
38372             this.html = '';
38373         }
38374         if(this.rendered){
38375             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38376      
38377             this.el.update(String.format(
38378                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38379                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38380             this.parentMenu.autoWidth();
38381         }
38382     },
38383
38384     // private
38385     handleClick : function(e){
38386         if(!this.href){ // if no link defined, stop the event automatically
38387             e.stopEvent();
38388         }
38389         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38390     },
38391
38392     // private
38393     activate : function(autoExpand){
38394         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38395             this.focus();
38396             if(autoExpand){
38397                 this.expandMenu();
38398             }
38399         }
38400         return true;
38401     },
38402
38403     // private
38404     shouldDeactivate : function(e){
38405         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38406             if(this.menu && this.menu.isVisible()){
38407                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38408             }
38409             return true;
38410         }
38411         return false;
38412     },
38413
38414     // private
38415     deactivate : function(){
38416         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38417         this.hideMenu();
38418     },
38419
38420     // private
38421     expandMenu : function(autoActivate){
38422         if(!this.disabled && this.menu){
38423             clearTimeout(this.hideTimer);
38424             delete this.hideTimer;
38425             if(!this.menu.isVisible() && !this.showTimer){
38426                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38427             }else if (this.menu.isVisible() && autoActivate){
38428                 this.menu.tryActivate(0, 1);
38429             }
38430         }
38431     },
38432
38433     // private
38434     deferExpand : function(autoActivate){
38435         delete this.showTimer;
38436         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38437         if(autoActivate){
38438             this.menu.tryActivate(0, 1);
38439         }
38440     },
38441
38442     // private
38443     hideMenu : function(){
38444         clearTimeout(this.showTimer);
38445         delete this.showTimer;
38446         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38447             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38448         }
38449     },
38450
38451     // private
38452     deferHide : function(){
38453         delete this.hideTimer;
38454         this.menu.hide();
38455     }
38456 });/*
38457  * Based on:
38458  * Ext JS Library 1.1.1
38459  * Copyright(c) 2006-2007, Ext JS, LLC.
38460  *
38461  * Originally Released Under LGPL - original licence link has changed is not relivant.
38462  *
38463  * Fork - LGPL
38464  * <script type="text/javascript">
38465  */
38466  
38467 /**
38468  * @class Roo.menu.CheckItem
38469  * @extends Roo.menu.Item
38470  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38471  * @constructor
38472  * Creates a new CheckItem
38473  * @param {Object} config Configuration options
38474  */
38475 Roo.menu.CheckItem = function(config){
38476     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38477     this.addEvents({
38478         /**
38479          * @event beforecheckchange
38480          * Fires before the checked value is set, providing an opportunity to cancel if needed
38481          * @param {Roo.menu.CheckItem} this
38482          * @param {Boolean} checked The new checked value that will be set
38483          */
38484         "beforecheckchange" : true,
38485         /**
38486          * @event checkchange
38487          * Fires after the checked value has been set
38488          * @param {Roo.menu.CheckItem} this
38489          * @param {Boolean} checked The checked value that was set
38490          */
38491         "checkchange" : true
38492     });
38493     if(this.checkHandler){
38494         this.on('checkchange', this.checkHandler, this.scope);
38495     }
38496 };
38497 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38498     /**
38499      * @cfg {String} group
38500      * All check items with the same group name will automatically be grouped into a single-select
38501      * radio button group (defaults to '')
38502      */
38503     /**
38504      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38505      */
38506     itemCls : "x-menu-item x-menu-check-item",
38507     /**
38508      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38509      */
38510     groupClass : "x-menu-group-item",
38511
38512     /**
38513      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38514      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38515      * initialized with checked = true will be rendered as checked.
38516      */
38517     checked: false,
38518
38519     // private
38520     ctype: "Roo.menu.CheckItem",
38521
38522     // private
38523     onRender : function(c){
38524         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38525         if(this.group){
38526             this.el.addClass(this.groupClass);
38527         }
38528         Roo.menu.MenuMgr.registerCheckable(this);
38529         if(this.checked){
38530             this.checked = false;
38531             this.setChecked(true, true);
38532         }
38533     },
38534
38535     // private
38536     destroy : function(){
38537         if(this.rendered){
38538             Roo.menu.MenuMgr.unregisterCheckable(this);
38539         }
38540         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38541     },
38542
38543     /**
38544      * Set the checked state of this item
38545      * @param {Boolean} checked The new checked value
38546      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38547      */
38548     setChecked : function(state, suppressEvent){
38549         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38550             if(this.container){
38551                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38552             }
38553             this.checked = state;
38554             if(suppressEvent !== true){
38555                 this.fireEvent("checkchange", this, state);
38556             }
38557         }
38558     },
38559
38560     // private
38561     handleClick : function(e){
38562        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38563            this.setChecked(!this.checked);
38564        }
38565        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38566     }
38567 });/*
38568  * Based on:
38569  * Ext JS Library 1.1.1
38570  * Copyright(c) 2006-2007, Ext JS, LLC.
38571  *
38572  * Originally Released Under LGPL - original licence link has changed is not relivant.
38573  *
38574  * Fork - LGPL
38575  * <script type="text/javascript">
38576  */
38577  
38578 /**
38579  * @class Roo.menu.DateItem
38580  * @extends Roo.menu.Adapter
38581  * A menu item that wraps the {@link Roo.DatPicker} component.
38582  * @constructor
38583  * Creates a new DateItem
38584  * @param {Object} config Configuration options
38585  */
38586 Roo.menu.DateItem = function(config){
38587     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38588     /** The Roo.DatePicker object @type Roo.DatePicker */
38589     this.picker = this.component;
38590     this.addEvents({select: true});
38591     
38592     this.picker.on("render", function(picker){
38593         picker.getEl().swallowEvent("click");
38594         picker.container.addClass("x-menu-date-item");
38595     });
38596
38597     this.picker.on("select", this.onSelect, this);
38598 };
38599
38600 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38601     // private
38602     onSelect : function(picker, date){
38603         this.fireEvent("select", this, date, picker);
38604         Roo.menu.DateItem.superclass.handleClick.call(this);
38605     }
38606 });/*
38607  * Based on:
38608  * Ext JS Library 1.1.1
38609  * Copyright(c) 2006-2007, Ext JS, LLC.
38610  *
38611  * Originally Released Under LGPL - original licence link has changed is not relivant.
38612  *
38613  * Fork - LGPL
38614  * <script type="text/javascript">
38615  */
38616  
38617 /**
38618  * @class Roo.menu.ColorItem
38619  * @extends Roo.menu.Adapter
38620  * A menu item that wraps the {@link Roo.ColorPalette} component.
38621  * @constructor
38622  * Creates a new ColorItem
38623  * @param {Object} config Configuration options
38624  */
38625 Roo.menu.ColorItem = function(config){
38626     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38627     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38628     this.palette = this.component;
38629     this.relayEvents(this.palette, ["select"]);
38630     if(this.selectHandler){
38631         this.on('select', this.selectHandler, this.scope);
38632     }
38633 };
38634 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38635  * Based on:
38636  * Ext JS Library 1.1.1
38637  * Copyright(c) 2006-2007, Ext JS, LLC.
38638  *
38639  * Originally Released Under LGPL - original licence link has changed is not relivant.
38640  *
38641  * Fork - LGPL
38642  * <script type="text/javascript">
38643  */
38644  
38645
38646 /**
38647  * @class Roo.menu.DateMenu
38648  * @extends Roo.menu.Menu
38649  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38650  * @constructor
38651  * Creates a new DateMenu
38652  * @param {Object} config Configuration options
38653  */
38654 Roo.menu.DateMenu = function(config){
38655     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38656     this.plain = true;
38657     var di = new Roo.menu.DateItem(config);
38658     this.add(di);
38659     /**
38660      * The {@link Roo.DatePicker} instance for this DateMenu
38661      * @type DatePicker
38662      */
38663     this.picker = di.picker;
38664     /**
38665      * @event select
38666      * @param {DatePicker} picker
38667      * @param {Date} date
38668      */
38669     this.relayEvents(di, ["select"]);
38670     this.on('beforeshow', function(){
38671         if(this.picker){
38672             this.picker.hideMonthPicker(false);
38673         }
38674     }, this);
38675 };
38676 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38677     cls:'x-date-menu'
38678 });/*
38679  * Based on:
38680  * Ext JS Library 1.1.1
38681  * Copyright(c) 2006-2007, Ext JS, LLC.
38682  *
38683  * Originally Released Under LGPL - original licence link has changed is not relivant.
38684  *
38685  * Fork - LGPL
38686  * <script type="text/javascript">
38687  */
38688  
38689
38690 /**
38691  * @class Roo.menu.ColorMenu
38692  * @extends Roo.menu.Menu
38693  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38694  * @constructor
38695  * Creates a new ColorMenu
38696  * @param {Object} config Configuration options
38697  */
38698 Roo.menu.ColorMenu = function(config){
38699     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38700     this.plain = true;
38701     var ci = new Roo.menu.ColorItem(config);
38702     this.add(ci);
38703     /**
38704      * The {@link Roo.ColorPalette} instance for this ColorMenu
38705      * @type ColorPalette
38706      */
38707     this.palette = ci.palette;
38708     /**
38709      * @event select
38710      * @param {ColorPalette} palette
38711      * @param {String} color
38712      */
38713     this.relayEvents(ci, ["select"]);
38714 };
38715 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38716  * Based on:
38717  * Ext JS Library 1.1.1
38718  * Copyright(c) 2006-2007, Ext JS, LLC.
38719  *
38720  * Originally Released Under LGPL - original licence link has changed is not relivant.
38721  *
38722  * Fork - LGPL
38723  * <script type="text/javascript">
38724  */
38725  
38726 /**
38727  * @class Roo.form.TextItem
38728  * @extends Roo.BoxComponent
38729  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38730  * @constructor
38731  * Creates a new TextItem
38732  * @param {Object} config Configuration options
38733  */
38734 Roo.form.TextItem = function(config){
38735     Roo.form.TextItem.superclass.constructor.call(this, config);
38736 };
38737
38738 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38739     
38740     /**
38741      * @cfg {String} tag the tag for this item (default div)
38742      */
38743     tag : 'div',
38744     /**
38745      * @cfg {String} html the content for this item
38746      */
38747     html : '',
38748     
38749     getAutoCreate : function()
38750     {
38751         var cfg = {
38752             id: this.id,
38753             tag: this.tag,
38754             html: this.html,
38755             cls: 'x-form-item'
38756         };
38757         
38758         return cfg;
38759         
38760     },
38761     
38762     onRender : function(ct, position)
38763     {
38764         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38765         
38766         if(!this.el){
38767             var cfg = this.getAutoCreate();
38768             if(!cfg.name){
38769                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38770             }
38771             if (!cfg.name.length) {
38772                 delete cfg.name;
38773             }
38774             this.el = ct.createChild(cfg, position);
38775         }
38776     }
38777     
38778 });/*
38779  * Based on:
38780  * Ext JS Library 1.1.1
38781  * Copyright(c) 2006-2007, Ext JS, LLC.
38782  *
38783  * Originally Released Under LGPL - original licence link has changed is not relivant.
38784  *
38785  * Fork - LGPL
38786  * <script type="text/javascript">
38787  */
38788  
38789 /**
38790  * @class Roo.form.Field
38791  * @extends Roo.BoxComponent
38792  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38793  * @constructor
38794  * Creates a new Field
38795  * @param {Object} config Configuration options
38796  */
38797 Roo.form.Field = function(config){
38798     Roo.form.Field.superclass.constructor.call(this, config);
38799 };
38800
38801 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38802     /**
38803      * @cfg {String} fieldLabel Label to use when rendering a form.
38804      */
38805        /**
38806      * @cfg {String} qtip Mouse over tip
38807      */
38808      
38809     /**
38810      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38811      */
38812     invalidClass : "x-form-invalid",
38813     /**
38814      * @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")
38815      */
38816     invalidText : "The value in this field is invalid",
38817     /**
38818      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38819      */
38820     focusClass : "x-form-focus",
38821     /**
38822      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38823       automatic validation (defaults to "keyup").
38824      */
38825     validationEvent : "keyup",
38826     /**
38827      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38828      */
38829     validateOnBlur : true,
38830     /**
38831      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38832      */
38833     validationDelay : 250,
38834     /**
38835      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38836      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38837      */
38838     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38839     /**
38840      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38841      */
38842     fieldClass : "x-form-field",
38843     /**
38844      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38845      *<pre>
38846 Value         Description
38847 -----------   ----------------------------------------------------------------------
38848 qtip          Display a quick tip when the user hovers over the field
38849 title         Display a default browser title attribute popup
38850 under         Add a block div beneath the field containing the error text
38851 side          Add an error icon to the right of the field with a popup on hover
38852 [element id]  Add the error text directly to the innerHTML of the specified element
38853 </pre>
38854      */
38855     msgTarget : 'qtip',
38856     /**
38857      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38858      */
38859     msgFx : 'normal',
38860
38861     /**
38862      * @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.
38863      */
38864     readOnly : false,
38865
38866     /**
38867      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38868      */
38869     disabled : false,
38870
38871     /**
38872      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38873      */
38874     inputType : undefined,
38875     
38876     /**
38877      * @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).
38878          */
38879         tabIndex : undefined,
38880         
38881     // private
38882     isFormField : true,
38883
38884     // private
38885     hasFocus : false,
38886     /**
38887      * @property {Roo.Element} fieldEl
38888      * Element Containing the rendered Field (with label etc.)
38889      */
38890     /**
38891      * @cfg {Mixed} value A value to initialize this field with.
38892      */
38893     value : undefined,
38894
38895     /**
38896      * @cfg {String} name The field's HTML name attribute.
38897      */
38898     /**
38899      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38900      */
38901     // private
38902     loadedValue : false,
38903      
38904      
38905         // private ??
38906         initComponent : function(){
38907         Roo.form.Field.superclass.initComponent.call(this);
38908         this.addEvents({
38909             /**
38910              * @event focus
38911              * Fires when this field receives input focus.
38912              * @param {Roo.form.Field} this
38913              */
38914             focus : true,
38915             /**
38916              * @event blur
38917              * Fires when this field loses input focus.
38918              * @param {Roo.form.Field} this
38919              */
38920             blur : true,
38921             /**
38922              * @event specialkey
38923              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38924              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38925              * @param {Roo.form.Field} this
38926              * @param {Roo.EventObject} e The event object
38927              */
38928             specialkey : true,
38929             /**
38930              * @event change
38931              * Fires just before the field blurs if the field value has changed.
38932              * @param {Roo.form.Field} this
38933              * @param {Mixed} newValue The new value
38934              * @param {Mixed} oldValue The original value
38935              */
38936             change : true,
38937             /**
38938              * @event invalid
38939              * Fires after the field has been marked as invalid.
38940              * @param {Roo.form.Field} this
38941              * @param {String} msg The validation message
38942              */
38943             invalid : true,
38944             /**
38945              * @event valid
38946              * Fires after the field has been validated with no errors.
38947              * @param {Roo.form.Field} this
38948              */
38949             valid : true,
38950              /**
38951              * @event keyup
38952              * Fires after the key up
38953              * @param {Roo.form.Field} this
38954              * @param {Roo.EventObject}  e The event Object
38955              */
38956             keyup : true
38957         });
38958     },
38959
38960     /**
38961      * Returns the name attribute of the field if available
38962      * @return {String} name The field name
38963      */
38964     getName: function(){
38965          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
38966     },
38967
38968     // private
38969     onRender : function(ct, position){
38970         Roo.form.Field.superclass.onRender.call(this, ct, position);
38971         if(!this.el){
38972             var cfg = this.getAutoCreate();
38973             if(!cfg.name){
38974                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38975             }
38976             if (!cfg.name.length) {
38977                 delete cfg.name;
38978             }
38979             if(this.inputType){
38980                 cfg.type = this.inputType;
38981             }
38982             this.el = ct.createChild(cfg, position);
38983         }
38984         var type = this.el.dom.type;
38985         if(type){
38986             if(type == 'password'){
38987                 type = 'text';
38988             }
38989             this.el.addClass('x-form-'+type);
38990         }
38991         if(this.readOnly){
38992             this.el.dom.readOnly = true;
38993         }
38994         if(this.tabIndex !== undefined){
38995             this.el.dom.setAttribute('tabIndex', this.tabIndex);
38996         }
38997
38998         this.el.addClass([this.fieldClass, this.cls]);
38999         this.initValue();
39000     },
39001
39002     /**
39003      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39004      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39005      * @return {Roo.form.Field} this
39006      */
39007     applyTo : function(target){
39008         this.allowDomMove = false;
39009         this.el = Roo.get(target);
39010         this.render(this.el.dom.parentNode);
39011         return this;
39012     },
39013
39014     // private
39015     initValue : function(){
39016         if(this.value !== undefined){
39017             this.setValue(this.value);
39018         }else if(this.el.dom.value.length > 0){
39019             this.setValue(this.el.dom.value);
39020         }
39021     },
39022
39023     /**
39024      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39025      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39026      */
39027     isDirty : function() {
39028         if(this.disabled) {
39029             return false;
39030         }
39031         return String(this.getValue()) !== String(this.originalValue);
39032     },
39033
39034     /**
39035      * stores the current value in loadedValue
39036      */
39037     resetHasChanged : function()
39038     {
39039         this.loadedValue = String(this.getValue());
39040     },
39041     /**
39042      * checks the current value against the 'loaded' value.
39043      * Note - will return false if 'resetHasChanged' has not been called first.
39044      */
39045     hasChanged : function()
39046     {
39047         if(this.disabled || this.readOnly) {
39048             return false;
39049         }
39050         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39051     },
39052     
39053     
39054     
39055     // private
39056     afterRender : function(){
39057         Roo.form.Field.superclass.afterRender.call(this);
39058         this.initEvents();
39059     },
39060
39061     // private
39062     fireKey : function(e){
39063         //Roo.log('field ' + e.getKey());
39064         if(e.isNavKeyPress()){
39065             this.fireEvent("specialkey", this, e);
39066         }
39067     },
39068
39069     /**
39070      * Resets the current field value to the originally loaded value and clears any validation messages
39071      */
39072     reset : function(){
39073         this.setValue(this.resetValue);
39074         this.originalValue = this.getValue();
39075         this.clearInvalid();
39076     },
39077
39078     // private
39079     initEvents : function(){
39080         // safari killled keypress - so keydown is now used..
39081         this.el.on("keydown" , this.fireKey,  this);
39082         this.el.on("focus", this.onFocus,  this);
39083         this.el.on("blur", this.onBlur,  this);
39084         this.el.relayEvent('keyup', this);
39085
39086         // reference to original value for reset
39087         this.originalValue = this.getValue();
39088         this.resetValue =  this.getValue();
39089     },
39090
39091     // private
39092     onFocus : function(){
39093         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39094             this.el.addClass(this.focusClass);
39095         }
39096         if(!this.hasFocus){
39097             this.hasFocus = true;
39098             this.startValue = this.getValue();
39099             this.fireEvent("focus", this);
39100         }
39101     },
39102
39103     beforeBlur : Roo.emptyFn,
39104
39105     // private
39106     onBlur : function(){
39107         this.beforeBlur();
39108         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39109             this.el.removeClass(this.focusClass);
39110         }
39111         this.hasFocus = false;
39112         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39113             this.validate();
39114         }
39115         var v = this.getValue();
39116         if(String(v) !== String(this.startValue)){
39117             this.fireEvent('change', this, v, this.startValue);
39118         }
39119         this.fireEvent("blur", this);
39120     },
39121
39122     /**
39123      * Returns whether or not the field value is currently valid
39124      * @param {Boolean} preventMark True to disable marking the field invalid
39125      * @return {Boolean} True if the value is valid, else false
39126      */
39127     isValid : function(preventMark){
39128         if(this.disabled){
39129             return true;
39130         }
39131         var restore = this.preventMark;
39132         this.preventMark = preventMark === true;
39133         var v = this.validateValue(this.processValue(this.getRawValue()));
39134         this.preventMark = restore;
39135         return v;
39136     },
39137
39138     /**
39139      * Validates the field value
39140      * @return {Boolean} True if the value is valid, else false
39141      */
39142     validate : function(){
39143         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39144             this.clearInvalid();
39145             return true;
39146         }
39147         return false;
39148     },
39149
39150     processValue : function(value){
39151         return value;
39152     },
39153
39154     // private
39155     // Subclasses should provide the validation implementation by overriding this
39156     validateValue : function(value){
39157         return true;
39158     },
39159
39160     /**
39161      * Mark this field as invalid
39162      * @param {String} msg The validation message
39163      */
39164     markInvalid : function(msg){
39165         if(!this.rendered || this.preventMark){ // not rendered
39166             return;
39167         }
39168         
39169         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39170         
39171         obj.el.addClass(this.invalidClass);
39172         msg = msg || this.invalidText;
39173         switch(this.msgTarget){
39174             case 'qtip':
39175                 obj.el.dom.qtip = msg;
39176                 obj.el.dom.qclass = 'x-form-invalid-tip';
39177                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39178                     Roo.QuickTips.enable();
39179                 }
39180                 break;
39181             case 'title':
39182                 this.el.dom.title = msg;
39183                 break;
39184             case 'under':
39185                 if(!this.errorEl){
39186                     var elp = this.el.findParent('.x-form-element', 5, true);
39187                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39188                     this.errorEl.setWidth(elp.getWidth(true)-20);
39189                 }
39190                 this.errorEl.update(msg);
39191                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39192                 break;
39193             case 'side':
39194                 if(!this.errorIcon){
39195                     var elp = this.el.findParent('.x-form-element', 5, true);
39196                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39197                 }
39198                 this.alignErrorIcon();
39199                 this.errorIcon.dom.qtip = msg;
39200                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39201                 this.errorIcon.show();
39202                 this.on('resize', this.alignErrorIcon, this);
39203                 break;
39204             default:
39205                 var t = Roo.getDom(this.msgTarget);
39206                 t.innerHTML = msg;
39207                 t.style.display = this.msgDisplay;
39208                 break;
39209         }
39210         this.fireEvent('invalid', this, msg);
39211     },
39212
39213     // private
39214     alignErrorIcon : function(){
39215         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39216     },
39217
39218     /**
39219      * Clear any invalid styles/messages for this field
39220      */
39221     clearInvalid : function(){
39222         if(!this.rendered || this.preventMark){ // not rendered
39223             return;
39224         }
39225         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39226         
39227         obj.el.removeClass(this.invalidClass);
39228         switch(this.msgTarget){
39229             case 'qtip':
39230                 obj.el.dom.qtip = '';
39231                 break;
39232             case 'title':
39233                 this.el.dom.title = '';
39234                 break;
39235             case 'under':
39236                 if(this.errorEl){
39237                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39238                 }
39239                 break;
39240             case 'side':
39241                 if(this.errorIcon){
39242                     this.errorIcon.dom.qtip = '';
39243                     this.errorIcon.hide();
39244                     this.un('resize', this.alignErrorIcon, this);
39245                 }
39246                 break;
39247             default:
39248                 var t = Roo.getDom(this.msgTarget);
39249                 t.innerHTML = '';
39250                 t.style.display = 'none';
39251                 break;
39252         }
39253         this.fireEvent('valid', this);
39254     },
39255
39256     /**
39257      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39258      * @return {Mixed} value The field value
39259      */
39260     getRawValue : function(){
39261         var v = this.el.getValue();
39262         
39263         return v;
39264     },
39265
39266     /**
39267      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39268      * @return {Mixed} value The field value
39269      */
39270     getValue : function(){
39271         var v = this.el.getValue();
39272          
39273         return v;
39274     },
39275
39276     /**
39277      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39278      * @param {Mixed} value The value to set
39279      */
39280     setRawValue : function(v){
39281         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39282     },
39283
39284     /**
39285      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39286      * @param {Mixed} value The value to set
39287      */
39288     setValue : function(v){
39289         this.value = v;
39290         if(this.rendered){
39291             this.el.dom.value = (v === null || v === undefined ? '' : v);
39292              this.validate();
39293         }
39294     },
39295
39296     adjustSize : function(w, h){
39297         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39298         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39299         return s;
39300     },
39301
39302     adjustWidth : function(tag, w){
39303         tag = tag.toLowerCase();
39304         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39305             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39306                 if(tag == 'input'){
39307                     return w + 2;
39308                 }
39309                 if(tag == 'textarea'){
39310                     return w-2;
39311                 }
39312             }else if(Roo.isOpera){
39313                 if(tag == 'input'){
39314                     return w + 2;
39315                 }
39316                 if(tag == 'textarea'){
39317                     return w-2;
39318                 }
39319             }
39320         }
39321         return w;
39322     }
39323 });
39324
39325
39326 // anything other than normal should be considered experimental
39327 Roo.form.Field.msgFx = {
39328     normal : {
39329         show: function(msgEl, f){
39330             msgEl.setDisplayed('block');
39331         },
39332
39333         hide : function(msgEl, f){
39334             msgEl.setDisplayed(false).update('');
39335         }
39336     },
39337
39338     slide : {
39339         show: function(msgEl, f){
39340             msgEl.slideIn('t', {stopFx:true});
39341         },
39342
39343         hide : function(msgEl, f){
39344             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39345         }
39346     },
39347
39348     slideRight : {
39349         show: function(msgEl, f){
39350             msgEl.fixDisplay();
39351             msgEl.alignTo(f.el, 'tl-tr');
39352             msgEl.slideIn('l', {stopFx:true});
39353         },
39354
39355         hide : function(msgEl, f){
39356             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39357         }
39358     }
39359 };/*
39360  * Based on:
39361  * Ext JS Library 1.1.1
39362  * Copyright(c) 2006-2007, Ext JS, LLC.
39363  *
39364  * Originally Released Under LGPL - original licence link has changed is not relivant.
39365  *
39366  * Fork - LGPL
39367  * <script type="text/javascript">
39368  */
39369  
39370
39371 /**
39372  * @class Roo.form.TextField
39373  * @extends Roo.form.Field
39374  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39375  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39376  * @constructor
39377  * Creates a new TextField
39378  * @param {Object} config Configuration options
39379  */
39380 Roo.form.TextField = function(config){
39381     Roo.form.TextField.superclass.constructor.call(this, config);
39382     this.addEvents({
39383         /**
39384          * @event autosize
39385          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39386          * according to the default logic, but this event provides a hook for the developer to apply additional
39387          * logic at runtime to resize the field if needed.
39388              * @param {Roo.form.Field} this This text field
39389              * @param {Number} width The new field width
39390              */
39391         autosize : true
39392     });
39393 };
39394
39395 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39396     /**
39397      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39398      */
39399     grow : false,
39400     /**
39401      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39402      */
39403     growMin : 30,
39404     /**
39405      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39406      */
39407     growMax : 800,
39408     /**
39409      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39410      */
39411     vtype : null,
39412     /**
39413      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39414      */
39415     maskRe : null,
39416     /**
39417      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39418      */
39419     disableKeyFilter : false,
39420     /**
39421      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39422      */
39423     allowBlank : true,
39424     /**
39425      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39426      */
39427     minLength : 0,
39428     /**
39429      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39430      */
39431     maxLength : Number.MAX_VALUE,
39432     /**
39433      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39434      */
39435     minLengthText : "The minimum length for this field is {0}",
39436     /**
39437      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39438      */
39439     maxLengthText : "The maximum length for this field is {0}",
39440     /**
39441      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39442      */
39443     selectOnFocus : false,
39444     /**
39445      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39446      */    
39447     allowLeadingSpace : false,
39448     /**
39449      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39450      */
39451     blankText : "This field is required",
39452     /**
39453      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39454      * If available, this function will be called only after the basic validators all return true, and will be passed the
39455      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39456      */
39457     validator : null,
39458     /**
39459      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39460      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39461      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39462      */
39463     regex : null,
39464     /**
39465      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39466      */
39467     regexText : "",
39468     /**
39469      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39470      */
39471     emptyText : null,
39472    
39473
39474     // private
39475     initEvents : function()
39476     {
39477         if (this.emptyText) {
39478             this.el.attr('placeholder', this.emptyText);
39479         }
39480         
39481         Roo.form.TextField.superclass.initEvents.call(this);
39482         if(this.validationEvent == 'keyup'){
39483             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39484             this.el.on('keyup', this.filterValidation, this);
39485         }
39486         else if(this.validationEvent !== false){
39487             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39488         }
39489         
39490         if(this.selectOnFocus){
39491             this.on("focus", this.preFocus, this);
39492         }
39493         if (!this.allowLeadingSpace) {
39494             this.on('blur', this.cleanLeadingSpace, this);
39495         }
39496         
39497         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39498             this.el.on("keypress", this.filterKeys, this);
39499         }
39500         if(this.grow){
39501             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39502             this.el.on("click", this.autoSize,  this);
39503         }
39504         if(this.el.is('input[type=password]') && Roo.isSafari){
39505             this.el.on('keydown', this.SafariOnKeyDown, this);
39506         }
39507     },
39508
39509     processValue : function(value){
39510         if(this.stripCharsRe){
39511             var newValue = value.replace(this.stripCharsRe, '');
39512             if(newValue !== value){
39513                 this.setRawValue(newValue);
39514                 return newValue;
39515             }
39516         }
39517         return value;
39518     },
39519
39520     filterValidation : function(e){
39521         if(!e.isNavKeyPress()){
39522             this.validationTask.delay(this.validationDelay);
39523         }
39524     },
39525
39526     // private
39527     onKeyUp : function(e){
39528         if(!e.isNavKeyPress()){
39529             this.autoSize();
39530         }
39531     },
39532     // private - clean the leading white space
39533     cleanLeadingSpace : function(e)
39534     {
39535         if ( this.inputType == 'file') {
39536             return;
39537         }
39538         
39539         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39540     },
39541     /**
39542      * Resets the current field value to the originally-loaded value and clears any validation messages.
39543      *  
39544      */
39545     reset : function(){
39546         Roo.form.TextField.superclass.reset.call(this);
39547        
39548     }, 
39549     // private
39550     preFocus : function(){
39551         
39552         if(this.selectOnFocus){
39553             this.el.dom.select();
39554         }
39555     },
39556
39557     
39558     // private
39559     filterKeys : function(e){
39560         var k = e.getKey();
39561         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39562             return;
39563         }
39564         var c = e.getCharCode(), cc = String.fromCharCode(c);
39565         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39566             return;
39567         }
39568         if(!this.maskRe.test(cc)){
39569             e.stopEvent();
39570         }
39571     },
39572
39573     setValue : function(v){
39574         
39575         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39576         
39577         this.autoSize();
39578     },
39579
39580     /**
39581      * Validates a value according to the field's validation rules and marks the field as invalid
39582      * if the validation fails
39583      * @param {Mixed} value The value to validate
39584      * @return {Boolean} True if the value is valid, else false
39585      */
39586     validateValue : function(value){
39587         if(value.length < 1)  { // if it's blank
39588              if(this.allowBlank){
39589                 this.clearInvalid();
39590                 return true;
39591              }else{
39592                 this.markInvalid(this.blankText);
39593                 return false;
39594              }
39595         }
39596         if(value.length < this.minLength){
39597             this.markInvalid(String.format(this.minLengthText, this.minLength));
39598             return false;
39599         }
39600         if(value.length > this.maxLength){
39601             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39602             return false;
39603         }
39604         if(this.vtype){
39605             var vt = Roo.form.VTypes;
39606             if(!vt[this.vtype](value, this)){
39607                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39608                 return false;
39609             }
39610         }
39611         if(typeof this.validator == "function"){
39612             var msg = this.validator(value);
39613             if(msg !== true){
39614                 this.markInvalid(msg);
39615                 return false;
39616             }
39617         }
39618         if(this.regex && !this.regex.test(value)){
39619             this.markInvalid(this.regexText);
39620             return false;
39621         }
39622         return true;
39623     },
39624
39625     /**
39626      * Selects text in this field
39627      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39628      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39629      */
39630     selectText : function(start, end){
39631         var v = this.getRawValue();
39632         if(v.length > 0){
39633             start = start === undefined ? 0 : start;
39634             end = end === undefined ? v.length : end;
39635             var d = this.el.dom;
39636             if(d.setSelectionRange){
39637                 d.setSelectionRange(start, end);
39638             }else if(d.createTextRange){
39639                 var range = d.createTextRange();
39640                 range.moveStart("character", start);
39641                 range.moveEnd("character", v.length-end);
39642                 range.select();
39643             }
39644         }
39645     },
39646
39647     /**
39648      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39649      * This only takes effect if grow = true, and fires the autosize event.
39650      */
39651     autoSize : function(){
39652         if(!this.grow || !this.rendered){
39653             return;
39654         }
39655         if(!this.metrics){
39656             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39657         }
39658         var el = this.el;
39659         var v = el.dom.value;
39660         var d = document.createElement('div');
39661         d.appendChild(document.createTextNode(v));
39662         v = d.innerHTML;
39663         d = null;
39664         v += "&#160;";
39665         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39666         this.el.setWidth(w);
39667         this.fireEvent("autosize", this, w);
39668     },
39669     
39670     // private
39671     SafariOnKeyDown : function(event)
39672     {
39673         // this is a workaround for a password hang bug on chrome/ webkit.
39674         
39675         var isSelectAll = false;
39676         
39677         if(this.el.dom.selectionEnd > 0){
39678             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39679         }
39680         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39681             event.preventDefault();
39682             this.setValue('');
39683             return;
39684         }
39685         
39686         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39687             
39688             event.preventDefault();
39689             // this is very hacky as keydown always get's upper case.
39690             
39691             var cc = String.fromCharCode(event.getCharCode());
39692             
39693             
39694             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39695             
39696         }
39697         
39698         
39699     }
39700 });/*
39701  * Based on:
39702  * Ext JS Library 1.1.1
39703  * Copyright(c) 2006-2007, Ext JS, LLC.
39704  *
39705  * Originally Released Under LGPL - original licence link has changed is not relivant.
39706  *
39707  * Fork - LGPL
39708  * <script type="text/javascript">
39709  */
39710  
39711 /**
39712  * @class Roo.form.Hidden
39713  * @extends Roo.form.TextField
39714  * Simple Hidden element used on forms 
39715  * 
39716  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39717  * 
39718  * @constructor
39719  * Creates a new Hidden form element.
39720  * @param {Object} config Configuration options
39721  */
39722
39723
39724
39725 // easy hidden field...
39726 Roo.form.Hidden = function(config){
39727     Roo.form.Hidden.superclass.constructor.call(this, config);
39728 };
39729   
39730 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39731     fieldLabel:      '',
39732     inputType:      'hidden',
39733     width:          50,
39734     allowBlank:     true,
39735     labelSeparator: '',
39736     hidden:         true,
39737     itemCls :       'x-form-item-display-none'
39738
39739
39740 });
39741
39742
39743 /*
39744  * Based on:
39745  * Ext JS Library 1.1.1
39746  * Copyright(c) 2006-2007, Ext JS, LLC.
39747  *
39748  * Originally Released Under LGPL - original licence link has changed is not relivant.
39749  *
39750  * Fork - LGPL
39751  * <script type="text/javascript">
39752  */
39753  
39754 /**
39755  * @class Roo.form.TriggerField
39756  * @extends Roo.form.TextField
39757  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39758  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39759  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39760  * for which you can provide a custom implementation.  For example:
39761  * <pre><code>
39762 var trigger = new Roo.form.TriggerField();
39763 trigger.onTriggerClick = myTriggerFn;
39764 trigger.applyTo('my-field');
39765 </code></pre>
39766  *
39767  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39768  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39769  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39770  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39771  * @constructor
39772  * Create a new TriggerField.
39773  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39774  * to the base TextField)
39775  */
39776 Roo.form.TriggerField = function(config){
39777     this.mimicing = false;
39778     Roo.form.TriggerField.superclass.constructor.call(this, config);
39779 };
39780
39781 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39782     /**
39783      * @cfg {String} triggerClass A CSS class to apply to the trigger
39784      */
39785     /**
39786      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39787      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39788      */
39789     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39790     /**
39791      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39792      */
39793     hideTrigger:false,
39794
39795     /** @cfg {Boolean} grow @hide */
39796     /** @cfg {Number} growMin @hide */
39797     /** @cfg {Number} growMax @hide */
39798
39799     /**
39800      * @hide 
39801      * @method
39802      */
39803     autoSize: Roo.emptyFn,
39804     // private
39805     monitorTab : true,
39806     // private
39807     deferHeight : true,
39808
39809     
39810     actionMode : 'wrap',
39811     // private
39812     onResize : function(w, h){
39813         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39814         if(typeof w == 'number'){
39815             var x = w - this.trigger.getWidth();
39816             this.el.setWidth(this.adjustWidth('input', x));
39817             this.trigger.setStyle('left', x+'px');
39818         }
39819     },
39820
39821     // private
39822     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39823
39824     // private
39825     getResizeEl : function(){
39826         return this.wrap;
39827     },
39828
39829     // private
39830     getPositionEl : function(){
39831         return this.wrap;
39832     },
39833
39834     // private
39835     alignErrorIcon : function(){
39836         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39837     },
39838
39839     // private
39840     onRender : function(ct, position){
39841         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39842         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39843         this.trigger = this.wrap.createChild(this.triggerConfig ||
39844                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39845         if(this.hideTrigger){
39846             this.trigger.setDisplayed(false);
39847         }
39848         this.initTrigger();
39849         if(!this.width){
39850             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39851         }
39852     },
39853
39854     // private
39855     initTrigger : function(){
39856         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39857         this.trigger.addClassOnOver('x-form-trigger-over');
39858         this.trigger.addClassOnClick('x-form-trigger-click');
39859     },
39860
39861     // private
39862     onDestroy : function(){
39863         if(this.trigger){
39864             this.trigger.removeAllListeners();
39865             this.trigger.remove();
39866         }
39867         if(this.wrap){
39868             this.wrap.remove();
39869         }
39870         Roo.form.TriggerField.superclass.onDestroy.call(this);
39871     },
39872
39873     // private
39874     onFocus : function(){
39875         Roo.form.TriggerField.superclass.onFocus.call(this);
39876         if(!this.mimicing){
39877             this.wrap.addClass('x-trigger-wrap-focus');
39878             this.mimicing = true;
39879             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39880             if(this.monitorTab){
39881                 this.el.on("keydown", this.checkTab, this);
39882             }
39883         }
39884     },
39885
39886     // private
39887     checkTab : function(e){
39888         if(e.getKey() == e.TAB){
39889             this.triggerBlur();
39890         }
39891     },
39892
39893     // private
39894     onBlur : function(){
39895         // do nothing
39896     },
39897
39898     // private
39899     mimicBlur : function(e, t){
39900         if(!this.wrap.contains(t) && this.validateBlur()){
39901             this.triggerBlur();
39902         }
39903     },
39904
39905     // private
39906     triggerBlur : function(){
39907         this.mimicing = false;
39908         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39909         if(this.monitorTab){
39910             this.el.un("keydown", this.checkTab, this);
39911         }
39912         this.wrap.removeClass('x-trigger-wrap-focus');
39913         Roo.form.TriggerField.superclass.onBlur.call(this);
39914     },
39915
39916     // private
39917     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39918     validateBlur : function(e, t){
39919         return true;
39920     },
39921
39922     // private
39923     onDisable : function(){
39924         Roo.form.TriggerField.superclass.onDisable.call(this);
39925         if(this.wrap){
39926             this.wrap.addClass('x-item-disabled');
39927         }
39928     },
39929
39930     // private
39931     onEnable : function(){
39932         Roo.form.TriggerField.superclass.onEnable.call(this);
39933         if(this.wrap){
39934             this.wrap.removeClass('x-item-disabled');
39935         }
39936     },
39937
39938     // private
39939     onShow : function(){
39940         var ae = this.getActionEl();
39941         
39942         if(ae){
39943             ae.dom.style.display = '';
39944             ae.dom.style.visibility = 'visible';
39945         }
39946     },
39947
39948     // private
39949     
39950     onHide : function(){
39951         var ae = this.getActionEl();
39952         ae.dom.style.display = 'none';
39953     },
39954
39955     /**
39956      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39957      * by an implementing function.
39958      * @method
39959      * @param {EventObject} e
39960      */
39961     onTriggerClick : Roo.emptyFn
39962 });
39963
39964 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
39965 // to be extended by an implementing class.  For an example of implementing this class, see the custom
39966 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
39967 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
39968     initComponent : function(){
39969         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
39970
39971         this.triggerConfig = {
39972             tag:'span', cls:'x-form-twin-triggers', cn:[
39973             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
39974             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
39975         ]};
39976     },
39977
39978     getTrigger : function(index){
39979         return this.triggers[index];
39980     },
39981
39982     initTrigger : function(){
39983         var ts = this.trigger.select('.x-form-trigger', true);
39984         this.wrap.setStyle('overflow', 'hidden');
39985         var triggerField = this;
39986         ts.each(function(t, all, index){
39987             t.hide = function(){
39988                 var w = triggerField.wrap.getWidth();
39989                 this.dom.style.display = 'none';
39990                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39991             };
39992             t.show = function(){
39993                 var w = triggerField.wrap.getWidth();
39994                 this.dom.style.display = '';
39995                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39996             };
39997             var triggerIndex = 'Trigger'+(index+1);
39998
39999             if(this['hide'+triggerIndex]){
40000                 t.dom.style.display = 'none';
40001             }
40002             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40003             t.addClassOnOver('x-form-trigger-over');
40004             t.addClassOnClick('x-form-trigger-click');
40005         }, this);
40006         this.triggers = ts.elements;
40007     },
40008
40009     onTrigger1Click : Roo.emptyFn,
40010     onTrigger2Click : Roo.emptyFn
40011 });/*
40012  * Based on:
40013  * Ext JS Library 1.1.1
40014  * Copyright(c) 2006-2007, Ext JS, LLC.
40015  *
40016  * Originally Released Under LGPL - original licence link has changed is not relivant.
40017  *
40018  * Fork - LGPL
40019  * <script type="text/javascript">
40020  */
40021  
40022 /**
40023  * @class Roo.form.TextArea
40024  * @extends Roo.form.TextField
40025  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40026  * support for auto-sizing.
40027  * @constructor
40028  * Creates a new TextArea
40029  * @param {Object} config Configuration options
40030  */
40031 Roo.form.TextArea = function(config){
40032     Roo.form.TextArea.superclass.constructor.call(this, config);
40033     // these are provided exchanges for backwards compat
40034     // minHeight/maxHeight were replaced by growMin/growMax to be
40035     // compatible with TextField growing config values
40036     if(this.minHeight !== undefined){
40037         this.growMin = this.minHeight;
40038     }
40039     if(this.maxHeight !== undefined){
40040         this.growMax = this.maxHeight;
40041     }
40042 };
40043
40044 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40045     /**
40046      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40047      */
40048     growMin : 60,
40049     /**
40050      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40051      */
40052     growMax: 1000,
40053     /**
40054      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40055      * in the field (equivalent to setting overflow: hidden, defaults to false)
40056      */
40057     preventScrollbars: false,
40058     /**
40059      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40060      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40061      */
40062
40063     // private
40064     onRender : function(ct, position){
40065         if(!this.el){
40066             this.defaultAutoCreate = {
40067                 tag: "textarea",
40068                 style:"width:300px;height:60px;",
40069                 autocomplete: "new-password"
40070             };
40071         }
40072         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40073         if(this.grow){
40074             this.textSizeEl = Roo.DomHelper.append(document.body, {
40075                 tag: "pre", cls: "x-form-grow-sizer"
40076             });
40077             if(this.preventScrollbars){
40078                 this.el.setStyle("overflow", "hidden");
40079             }
40080             this.el.setHeight(this.growMin);
40081         }
40082     },
40083
40084     onDestroy : function(){
40085         if(this.textSizeEl){
40086             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40087         }
40088         Roo.form.TextArea.superclass.onDestroy.call(this);
40089     },
40090
40091     // private
40092     onKeyUp : function(e){
40093         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40094             this.autoSize();
40095         }
40096     },
40097
40098     /**
40099      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40100      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40101      */
40102     autoSize : function(){
40103         if(!this.grow || !this.textSizeEl){
40104             return;
40105         }
40106         var el = this.el;
40107         var v = el.dom.value;
40108         var ts = this.textSizeEl;
40109
40110         ts.innerHTML = '';
40111         ts.appendChild(document.createTextNode(v));
40112         v = ts.innerHTML;
40113
40114         Roo.fly(ts).setWidth(this.el.getWidth());
40115         if(v.length < 1){
40116             v = "&#160;&#160;";
40117         }else{
40118             if(Roo.isIE){
40119                 v = v.replace(/\n/g, '<p>&#160;</p>');
40120             }
40121             v += "&#160;\n&#160;";
40122         }
40123         ts.innerHTML = v;
40124         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40125         if(h != this.lastHeight){
40126             this.lastHeight = h;
40127             this.el.setHeight(h);
40128             this.fireEvent("autosize", this, h);
40129         }
40130     }
40131 });/*
40132  * Based on:
40133  * Ext JS Library 1.1.1
40134  * Copyright(c) 2006-2007, Ext JS, LLC.
40135  *
40136  * Originally Released Under LGPL - original licence link has changed is not relivant.
40137  *
40138  * Fork - LGPL
40139  * <script type="text/javascript">
40140  */
40141  
40142
40143 /**
40144  * @class Roo.form.NumberField
40145  * @extends Roo.form.TextField
40146  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40147  * @constructor
40148  * Creates a new NumberField
40149  * @param {Object} config Configuration options
40150  */
40151 Roo.form.NumberField = function(config){
40152     Roo.form.NumberField.superclass.constructor.call(this, config);
40153 };
40154
40155 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40156     /**
40157      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40158      */
40159     fieldClass: "x-form-field x-form-num-field",
40160     /**
40161      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40162      */
40163     allowDecimals : true,
40164     /**
40165      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40166      */
40167     decimalSeparator : ".",
40168     /**
40169      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40170      */
40171     decimalPrecision : 2,
40172     /**
40173      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40174      */
40175     allowNegative : true,
40176     /**
40177      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40178      */
40179     minValue : Number.NEGATIVE_INFINITY,
40180     /**
40181      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40182      */
40183     maxValue : Number.MAX_VALUE,
40184     /**
40185      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40186      */
40187     minText : "The minimum value for this field is {0}",
40188     /**
40189      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40190      */
40191     maxText : "The maximum value for this field is {0}",
40192     /**
40193      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40194      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40195      */
40196     nanText : "{0} is not a valid number",
40197
40198     // private
40199     initEvents : function(){
40200         Roo.form.NumberField.superclass.initEvents.call(this);
40201         var allowed = "0123456789";
40202         if(this.allowDecimals){
40203             allowed += this.decimalSeparator;
40204         }
40205         if(this.allowNegative){
40206             allowed += "-";
40207         }
40208         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40209         var keyPress = function(e){
40210             var k = e.getKey();
40211             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40212                 return;
40213             }
40214             var c = e.getCharCode();
40215             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40216                 e.stopEvent();
40217             }
40218         };
40219         this.el.on("keypress", keyPress, this);
40220     },
40221
40222     // private
40223     validateValue : function(value){
40224         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40225             return false;
40226         }
40227         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40228              return true;
40229         }
40230         var num = this.parseValue(value);
40231         if(isNaN(num)){
40232             this.markInvalid(String.format(this.nanText, value));
40233             return false;
40234         }
40235         if(num < this.minValue){
40236             this.markInvalid(String.format(this.minText, this.minValue));
40237             return false;
40238         }
40239         if(num > this.maxValue){
40240             this.markInvalid(String.format(this.maxText, this.maxValue));
40241             return false;
40242         }
40243         return true;
40244     },
40245
40246     getValue : function(){
40247         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40248     },
40249
40250     // private
40251     parseValue : function(value){
40252         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40253         return isNaN(value) ? '' : value;
40254     },
40255
40256     // private
40257     fixPrecision : function(value){
40258         var nan = isNaN(value);
40259         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40260             return nan ? '' : value;
40261         }
40262         return parseFloat(value).toFixed(this.decimalPrecision);
40263     },
40264
40265     setValue : function(v){
40266         v = this.fixPrecision(v);
40267         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40268     },
40269
40270     // private
40271     decimalPrecisionFcn : function(v){
40272         return Math.floor(v);
40273     },
40274
40275     beforeBlur : function(){
40276         var v = this.parseValue(this.getRawValue());
40277         if(v){
40278             this.setValue(v);
40279         }
40280     }
40281 });/*
40282  * Based on:
40283  * Ext JS Library 1.1.1
40284  * Copyright(c) 2006-2007, Ext JS, LLC.
40285  *
40286  * Originally Released Under LGPL - original licence link has changed is not relivant.
40287  *
40288  * Fork - LGPL
40289  * <script type="text/javascript">
40290  */
40291  
40292 /**
40293  * @class Roo.form.DateField
40294  * @extends Roo.form.TriggerField
40295  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40296 * @constructor
40297 * Create a new DateField
40298 * @param {Object} config
40299  */
40300 Roo.form.DateField = function(config)
40301 {
40302     Roo.form.DateField.superclass.constructor.call(this, config);
40303     
40304       this.addEvents({
40305          
40306         /**
40307          * @event select
40308          * Fires when a date is selected
40309              * @param {Roo.form.DateField} combo This combo box
40310              * @param {Date} date The date selected
40311              */
40312         'select' : true
40313          
40314     });
40315     
40316     
40317     if(typeof this.minValue == "string") {
40318         this.minValue = this.parseDate(this.minValue);
40319     }
40320     if(typeof this.maxValue == "string") {
40321         this.maxValue = this.parseDate(this.maxValue);
40322     }
40323     this.ddMatch = null;
40324     if(this.disabledDates){
40325         var dd = this.disabledDates;
40326         var re = "(?:";
40327         for(var i = 0; i < dd.length; i++){
40328             re += dd[i];
40329             if(i != dd.length-1) {
40330                 re += "|";
40331             }
40332         }
40333         this.ddMatch = new RegExp(re + ")");
40334     }
40335 };
40336
40337 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40338     /**
40339      * @cfg {String} format
40340      * The default date format string which can be overriden for localization support.  The format must be
40341      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40342      */
40343     format : "m/d/y",
40344     /**
40345      * @cfg {String} altFormats
40346      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40347      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40348      */
40349     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40350     /**
40351      * @cfg {Array} disabledDays
40352      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40353      */
40354     disabledDays : null,
40355     /**
40356      * @cfg {String} disabledDaysText
40357      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40358      */
40359     disabledDaysText : "Disabled",
40360     /**
40361      * @cfg {Array} disabledDates
40362      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40363      * expression so they are very powerful. Some examples:
40364      * <ul>
40365      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40366      * <li>["03/08", "09/16"] would disable those days for every year</li>
40367      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40368      * <li>["03/../2006"] would disable every day in March 2006</li>
40369      * <li>["^03"] would disable every day in every March</li>
40370      * </ul>
40371      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40372      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40373      */
40374     disabledDates : null,
40375     /**
40376      * @cfg {String} disabledDatesText
40377      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40378      */
40379     disabledDatesText : "Disabled",
40380     /**
40381      * @cfg {Date/String} minValue
40382      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40383      * valid format (defaults to null).
40384      */
40385     minValue : null,
40386     /**
40387      * @cfg {Date/String} maxValue
40388      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40389      * valid format (defaults to null).
40390      */
40391     maxValue : null,
40392     /**
40393      * @cfg {String} minText
40394      * The error text to display when the date in the cell is before minValue (defaults to
40395      * 'The date in this field must be after {minValue}').
40396      */
40397     minText : "The date in this field must be equal to or after {0}",
40398     /**
40399      * @cfg {String} maxText
40400      * The error text to display when the date in the cell is after maxValue (defaults to
40401      * 'The date in this field must be before {maxValue}').
40402      */
40403     maxText : "The date in this field must be equal to or before {0}",
40404     /**
40405      * @cfg {String} invalidText
40406      * The error text to display when the date in the field is invalid (defaults to
40407      * '{value} is not a valid date - it must be in the format {format}').
40408      */
40409     invalidText : "{0} is not a valid date - it must be in the format {1}",
40410     /**
40411      * @cfg {String} triggerClass
40412      * An additional CSS class used to style the trigger button.  The trigger will always get the
40413      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40414      * which displays a calendar icon).
40415      */
40416     triggerClass : 'x-form-date-trigger',
40417     
40418
40419     /**
40420      * @cfg {Boolean} useIso
40421      * if enabled, then the date field will use a hidden field to store the 
40422      * real value as iso formated date. default (false)
40423      */ 
40424     useIso : false,
40425     /**
40426      * @cfg {String/Object} autoCreate
40427      * A DomHelper element spec, or true for a default element spec (defaults to
40428      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40429      */ 
40430     // private
40431     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40432     
40433     // private
40434     hiddenField: false,
40435     
40436     onRender : function(ct, position)
40437     {
40438         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40439         if (this.useIso) {
40440             //this.el.dom.removeAttribute('name'); 
40441             Roo.log("Changing name?");
40442             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40443             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40444                     'before', true);
40445             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40446             // prevent input submission
40447             this.hiddenName = this.name;
40448         }
40449             
40450             
40451     },
40452     
40453     // private
40454     validateValue : function(value)
40455     {
40456         value = this.formatDate(value);
40457         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40458             Roo.log('super failed');
40459             return false;
40460         }
40461         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40462              return true;
40463         }
40464         var svalue = value;
40465         value = this.parseDate(value);
40466         if(!value){
40467             Roo.log('parse date failed' + svalue);
40468             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40469             return false;
40470         }
40471         var time = value.getTime();
40472         if(this.minValue && time < this.minValue.getTime()){
40473             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40474             return false;
40475         }
40476         if(this.maxValue && time > this.maxValue.getTime()){
40477             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40478             return false;
40479         }
40480         if(this.disabledDays){
40481             var day = value.getDay();
40482             for(var i = 0; i < this.disabledDays.length; i++) {
40483                 if(day === this.disabledDays[i]){
40484                     this.markInvalid(this.disabledDaysText);
40485                     return false;
40486                 }
40487             }
40488         }
40489         var fvalue = this.formatDate(value);
40490         if(this.ddMatch && this.ddMatch.test(fvalue)){
40491             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40492             return false;
40493         }
40494         return true;
40495     },
40496
40497     // private
40498     // Provides logic to override the default TriggerField.validateBlur which just returns true
40499     validateBlur : function(){
40500         return !this.menu || !this.menu.isVisible();
40501     },
40502     
40503     getName: function()
40504     {
40505         // returns hidden if it's set..
40506         if (!this.rendered) {return ''};
40507         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40508         
40509     },
40510
40511     /**
40512      * Returns the current date value of the date field.
40513      * @return {Date} The date value
40514      */
40515     getValue : function(){
40516         
40517         return  this.hiddenField ?
40518                 this.hiddenField.value :
40519                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40520     },
40521
40522     /**
40523      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40524      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40525      * (the default format used is "m/d/y").
40526      * <br />Usage:
40527      * <pre><code>
40528 //All of these calls set the same date value (May 4, 2006)
40529
40530 //Pass a date object:
40531 var dt = new Date('5/4/06');
40532 dateField.setValue(dt);
40533
40534 //Pass a date string (default format):
40535 dateField.setValue('5/4/06');
40536
40537 //Pass a date string (custom format):
40538 dateField.format = 'Y-m-d';
40539 dateField.setValue('2006-5-4');
40540 </code></pre>
40541      * @param {String/Date} date The date or valid date string
40542      */
40543     setValue : function(date){
40544         if (this.hiddenField) {
40545             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40546         }
40547         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40548         // make sure the value field is always stored as a date..
40549         this.value = this.parseDate(date);
40550         
40551         
40552     },
40553
40554     // private
40555     parseDate : function(value){
40556         if(!value || value instanceof Date){
40557             return value;
40558         }
40559         var v = Date.parseDate(value, this.format);
40560          if (!v && this.useIso) {
40561             v = Date.parseDate(value, 'Y-m-d');
40562         }
40563         if(!v && this.altFormats){
40564             if(!this.altFormatsArray){
40565                 this.altFormatsArray = this.altFormats.split("|");
40566             }
40567             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40568                 v = Date.parseDate(value, this.altFormatsArray[i]);
40569             }
40570         }
40571         return v;
40572     },
40573
40574     // private
40575     formatDate : function(date, fmt){
40576         return (!date || !(date instanceof Date)) ?
40577                date : date.dateFormat(fmt || this.format);
40578     },
40579
40580     // private
40581     menuListeners : {
40582         select: function(m, d){
40583             
40584             this.setValue(d);
40585             this.fireEvent('select', this, d);
40586         },
40587         show : function(){ // retain focus styling
40588             this.onFocus();
40589         },
40590         hide : function(){
40591             this.focus.defer(10, this);
40592             var ml = this.menuListeners;
40593             this.menu.un("select", ml.select,  this);
40594             this.menu.un("show", ml.show,  this);
40595             this.menu.un("hide", ml.hide,  this);
40596         }
40597     },
40598
40599     // private
40600     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40601     onTriggerClick : function(){
40602         if(this.disabled){
40603             return;
40604         }
40605         if(this.menu == null){
40606             this.menu = new Roo.menu.DateMenu();
40607         }
40608         Roo.apply(this.menu.picker,  {
40609             showClear: this.allowBlank,
40610             minDate : this.minValue,
40611             maxDate : this.maxValue,
40612             disabledDatesRE : this.ddMatch,
40613             disabledDatesText : this.disabledDatesText,
40614             disabledDays : this.disabledDays,
40615             disabledDaysText : this.disabledDaysText,
40616             format : this.useIso ? 'Y-m-d' : this.format,
40617             minText : String.format(this.minText, this.formatDate(this.minValue)),
40618             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40619         });
40620         this.menu.on(Roo.apply({}, this.menuListeners, {
40621             scope:this
40622         }));
40623         this.menu.picker.setValue(this.getValue() || new Date());
40624         this.menu.show(this.el, "tl-bl?");
40625     },
40626
40627     beforeBlur : function(){
40628         var v = this.parseDate(this.getRawValue());
40629         if(v){
40630             this.setValue(v);
40631         }
40632     },
40633
40634     /*@
40635      * overide
40636      * 
40637      */
40638     isDirty : function() {
40639         if(this.disabled) {
40640             return false;
40641         }
40642         
40643         if(typeof(this.startValue) === 'undefined'){
40644             return false;
40645         }
40646         
40647         return String(this.getValue()) !== String(this.startValue);
40648         
40649     },
40650     // @overide
40651     cleanLeadingSpace : function(e)
40652     {
40653        return;
40654     }
40655     
40656 });/*
40657  * Based on:
40658  * Ext JS Library 1.1.1
40659  * Copyright(c) 2006-2007, Ext JS, LLC.
40660  *
40661  * Originally Released Under LGPL - original licence link has changed is not relivant.
40662  *
40663  * Fork - LGPL
40664  * <script type="text/javascript">
40665  */
40666  
40667 /**
40668  * @class Roo.form.MonthField
40669  * @extends Roo.form.TriggerField
40670  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40671 * @constructor
40672 * Create a new MonthField
40673 * @param {Object} config
40674  */
40675 Roo.form.MonthField = function(config){
40676     
40677     Roo.form.MonthField.superclass.constructor.call(this, config);
40678     
40679       this.addEvents({
40680          
40681         /**
40682          * @event select
40683          * Fires when a date is selected
40684              * @param {Roo.form.MonthFieeld} combo This combo box
40685              * @param {Date} date The date selected
40686              */
40687         'select' : true
40688          
40689     });
40690     
40691     
40692     if(typeof this.minValue == "string") {
40693         this.minValue = this.parseDate(this.minValue);
40694     }
40695     if(typeof this.maxValue == "string") {
40696         this.maxValue = this.parseDate(this.maxValue);
40697     }
40698     this.ddMatch = null;
40699     if(this.disabledDates){
40700         var dd = this.disabledDates;
40701         var re = "(?:";
40702         for(var i = 0; i < dd.length; i++){
40703             re += dd[i];
40704             if(i != dd.length-1) {
40705                 re += "|";
40706             }
40707         }
40708         this.ddMatch = new RegExp(re + ")");
40709     }
40710 };
40711
40712 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40713     /**
40714      * @cfg {String} format
40715      * The default date format string which can be overriden for localization support.  The format must be
40716      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40717      */
40718     format : "M Y",
40719     /**
40720      * @cfg {String} altFormats
40721      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40722      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40723      */
40724     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40725     /**
40726      * @cfg {Array} disabledDays
40727      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40728      */
40729     disabledDays : [0,1,2,3,4,5,6],
40730     /**
40731      * @cfg {String} disabledDaysText
40732      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40733      */
40734     disabledDaysText : "Disabled",
40735     /**
40736      * @cfg {Array} disabledDates
40737      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40738      * expression so they are very powerful. Some examples:
40739      * <ul>
40740      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40741      * <li>["03/08", "09/16"] would disable those days for every year</li>
40742      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40743      * <li>["03/../2006"] would disable every day in March 2006</li>
40744      * <li>["^03"] would disable every day in every March</li>
40745      * </ul>
40746      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40747      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40748      */
40749     disabledDates : null,
40750     /**
40751      * @cfg {String} disabledDatesText
40752      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40753      */
40754     disabledDatesText : "Disabled",
40755     /**
40756      * @cfg {Date/String} minValue
40757      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40758      * valid format (defaults to null).
40759      */
40760     minValue : null,
40761     /**
40762      * @cfg {Date/String} maxValue
40763      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40764      * valid format (defaults to null).
40765      */
40766     maxValue : null,
40767     /**
40768      * @cfg {String} minText
40769      * The error text to display when the date in the cell is before minValue (defaults to
40770      * 'The date in this field must be after {minValue}').
40771      */
40772     minText : "The date in this field must be equal to or after {0}",
40773     /**
40774      * @cfg {String} maxTextf
40775      * The error text to display when the date in the cell is after maxValue (defaults to
40776      * 'The date in this field must be before {maxValue}').
40777      */
40778     maxText : "The date in this field must be equal to or before {0}",
40779     /**
40780      * @cfg {String} invalidText
40781      * The error text to display when the date in the field is invalid (defaults to
40782      * '{value} is not a valid date - it must be in the format {format}').
40783      */
40784     invalidText : "{0} is not a valid date - it must be in the format {1}",
40785     /**
40786      * @cfg {String} triggerClass
40787      * An additional CSS class used to style the trigger button.  The trigger will always get the
40788      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40789      * which displays a calendar icon).
40790      */
40791     triggerClass : 'x-form-date-trigger',
40792     
40793
40794     /**
40795      * @cfg {Boolean} useIso
40796      * if enabled, then the date field will use a hidden field to store the 
40797      * real value as iso formated date. default (true)
40798      */ 
40799     useIso : true,
40800     /**
40801      * @cfg {String/Object} autoCreate
40802      * A DomHelper element spec, or true for a default element spec (defaults to
40803      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40804      */ 
40805     // private
40806     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40807     
40808     // private
40809     hiddenField: false,
40810     
40811     hideMonthPicker : false,
40812     
40813     onRender : function(ct, position)
40814     {
40815         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40816         if (this.useIso) {
40817             this.el.dom.removeAttribute('name'); 
40818             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40819                     'before', true);
40820             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40821             // prevent input submission
40822             this.hiddenName = this.name;
40823         }
40824             
40825             
40826     },
40827     
40828     // private
40829     validateValue : function(value)
40830     {
40831         value = this.formatDate(value);
40832         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40833             return false;
40834         }
40835         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40836              return true;
40837         }
40838         var svalue = value;
40839         value = this.parseDate(value);
40840         if(!value){
40841             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40842             return false;
40843         }
40844         var time = value.getTime();
40845         if(this.minValue && time < this.minValue.getTime()){
40846             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40847             return false;
40848         }
40849         if(this.maxValue && time > this.maxValue.getTime()){
40850             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40851             return false;
40852         }
40853         /*if(this.disabledDays){
40854             var day = value.getDay();
40855             for(var i = 0; i < this.disabledDays.length; i++) {
40856                 if(day === this.disabledDays[i]){
40857                     this.markInvalid(this.disabledDaysText);
40858                     return false;
40859                 }
40860             }
40861         }
40862         */
40863         var fvalue = this.formatDate(value);
40864         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40865             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40866             return false;
40867         }
40868         */
40869         return true;
40870     },
40871
40872     // private
40873     // Provides logic to override the default TriggerField.validateBlur which just returns true
40874     validateBlur : function(){
40875         return !this.menu || !this.menu.isVisible();
40876     },
40877
40878     /**
40879      * Returns the current date value of the date field.
40880      * @return {Date} The date value
40881      */
40882     getValue : function(){
40883         
40884         
40885         
40886         return  this.hiddenField ?
40887                 this.hiddenField.value :
40888                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40889     },
40890
40891     /**
40892      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40893      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40894      * (the default format used is "m/d/y").
40895      * <br />Usage:
40896      * <pre><code>
40897 //All of these calls set the same date value (May 4, 2006)
40898
40899 //Pass a date object:
40900 var dt = new Date('5/4/06');
40901 monthField.setValue(dt);
40902
40903 //Pass a date string (default format):
40904 monthField.setValue('5/4/06');
40905
40906 //Pass a date string (custom format):
40907 monthField.format = 'Y-m-d';
40908 monthField.setValue('2006-5-4');
40909 </code></pre>
40910      * @param {String/Date} date The date or valid date string
40911      */
40912     setValue : function(date){
40913         Roo.log('month setValue' + date);
40914         // can only be first of month..
40915         
40916         var val = this.parseDate(date);
40917         
40918         if (this.hiddenField) {
40919             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40920         }
40921         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40922         this.value = this.parseDate(date);
40923     },
40924
40925     // private
40926     parseDate : function(value){
40927         if(!value || value instanceof Date){
40928             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40929             return value;
40930         }
40931         var v = Date.parseDate(value, this.format);
40932         if (!v && this.useIso) {
40933             v = Date.parseDate(value, 'Y-m-d');
40934         }
40935         if (v) {
40936             // 
40937             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40938         }
40939         
40940         
40941         if(!v && this.altFormats){
40942             if(!this.altFormatsArray){
40943                 this.altFormatsArray = this.altFormats.split("|");
40944             }
40945             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40946                 v = Date.parseDate(value, this.altFormatsArray[i]);
40947             }
40948         }
40949         return v;
40950     },
40951
40952     // private
40953     formatDate : function(date, fmt){
40954         return (!date || !(date instanceof Date)) ?
40955                date : date.dateFormat(fmt || this.format);
40956     },
40957
40958     // private
40959     menuListeners : {
40960         select: function(m, d){
40961             this.setValue(d);
40962             this.fireEvent('select', this, d);
40963         },
40964         show : function(){ // retain focus styling
40965             this.onFocus();
40966         },
40967         hide : function(){
40968             this.focus.defer(10, this);
40969             var ml = this.menuListeners;
40970             this.menu.un("select", ml.select,  this);
40971             this.menu.un("show", ml.show,  this);
40972             this.menu.un("hide", ml.hide,  this);
40973         }
40974     },
40975     // private
40976     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40977     onTriggerClick : function(){
40978         if(this.disabled){
40979             return;
40980         }
40981         if(this.menu == null){
40982             this.menu = new Roo.menu.DateMenu();
40983            
40984         }
40985         
40986         Roo.apply(this.menu.picker,  {
40987             
40988             showClear: this.allowBlank,
40989             minDate : this.minValue,
40990             maxDate : this.maxValue,
40991             disabledDatesRE : this.ddMatch,
40992             disabledDatesText : this.disabledDatesText,
40993             
40994             format : this.useIso ? 'Y-m-d' : this.format,
40995             minText : String.format(this.minText, this.formatDate(this.minValue)),
40996             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40997             
40998         });
40999          this.menu.on(Roo.apply({}, this.menuListeners, {
41000             scope:this
41001         }));
41002        
41003         
41004         var m = this.menu;
41005         var p = m.picker;
41006         
41007         // hide month picker get's called when we called by 'before hide';
41008         
41009         var ignorehide = true;
41010         p.hideMonthPicker  = function(disableAnim){
41011             if (ignorehide) {
41012                 return;
41013             }
41014              if(this.monthPicker){
41015                 Roo.log("hideMonthPicker called");
41016                 if(disableAnim === true){
41017                     this.monthPicker.hide();
41018                 }else{
41019                     this.monthPicker.slideOut('t', {duration:.2});
41020                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41021                     p.fireEvent("select", this, this.value);
41022                     m.hide();
41023                 }
41024             }
41025         }
41026         
41027         Roo.log('picker set value');
41028         Roo.log(this.getValue());
41029         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41030         m.show(this.el, 'tl-bl?');
41031         ignorehide  = false;
41032         // this will trigger hideMonthPicker..
41033         
41034         
41035         // hidden the day picker
41036         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41037         
41038         
41039         
41040       
41041         
41042         p.showMonthPicker.defer(100, p);
41043     
41044         
41045        
41046     },
41047
41048     beforeBlur : function(){
41049         var v = this.parseDate(this.getRawValue());
41050         if(v){
41051             this.setValue(v);
41052         }
41053     }
41054
41055     /** @cfg {Boolean} grow @hide */
41056     /** @cfg {Number} growMin @hide */
41057     /** @cfg {Number} growMax @hide */
41058     /**
41059      * @hide
41060      * @method autoSize
41061      */
41062 });/*
41063  * Based on:
41064  * Ext JS Library 1.1.1
41065  * Copyright(c) 2006-2007, Ext JS, LLC.
41066  *
41067  * Originally Released Under LGPL - original licence link has changed is not relivant.
41068  *
41069  * Fork - LGPL
41070  * <script type="text/javascript">
41071  */
41072  
41073
41074 /**
41075  * @class Roo.form.ComboBox
41076  * @extends Roo.form.TriggerField
41077  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41078  * @constructor
41079  * Create a new ComboBox.
41080  * @param {Object} config Configuration options
41081  */
41082 Roo.form.ComboBox = function(config){
41083     Roo.form.ComboBox.superclass.constructor.call(this, config);
41084     this.addEvents({
41085         /**
41086          * @event expand
41087          * Fires when the dropdown list is expanded
41088              * @param {Roo.form.ComboBox} combo This combo box
41089              */
41090         'expand' : true,
41091         /**
41092          * @event collapse
41093          * Fires when the dropdown list is collapsed
41094              * @param {Roo.form.ComboBox} combo This combo box
41095              */
41096         'collapse' : true,
41097         /**
41098          * @event beforeselect
41099          * Fires before a list item is selected. Return false to cancel the selection.
41100              * @param {Roo.form.ComboBox} combo This combo box
41101              * @param {Roo.data.Record} record The data record returned from the underlying store
41102              * @param {Number} index The index of the selected item in the dropdown list
41103              */
41104         'beforeselect' : true,
41105         /**
41106          * @event select
41107          * Fires when a list item is selected
41108              * @param {Roo.form.ComboBox} combo This combo box
41109              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41110              * @param {Number} index The index of the selected item in the dropdown list
41111              */
41112         'select' : true,
41113         /**
41114          * @event beforequery
41115          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41116          * The event object passed has these properties:
41117              * @param {Roo.form.ComboBox} combo This combo box
41118              * @param {String} query The query
41119              * @param {Boolean} forceAll true to force "all" query
41120              * @param {Boolean} cancel true to cancel the query
41121              * @param {Object} e The query event object
41122              */
41123         'beforequery': true,
41124          /**
41125          * @event add
41126          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41127              * @param {Roo.form.ComboBox} combo This combo box
41128              */
41129         'add' : true,
41130         /**
41131          * @event edit
41132          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41133              * @param {Roo.form.ComboBox} combo This combo box
41134              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41135              */
41136         'edit' : true
41137         
41138         
41139     });
41140     if(this.transform){
41141         this.allowDomMove = false;
41142         var s = Roo.getDom(this.transform);
41143         if(!this.hiddenName){
41144             this.hiddenName = s.name;
41145         }
41146         if(!this.store){
41147             this.mode = 'local';
41148             var d = [], opts = s.options;
41149             for(var i = 0, len = opts.length;i < len; i++){
41150                 var o = opts[i];
41151                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41152                 if(o.selected) {
41153                     this.value = value;
41154                 }
41155                 d.push([value, o.text]);
41156             }
41157             this.store = new Roo.data.SimpleStore({
41158                 'id': 0,
41159                 fields: ['value', 'text'],
41160                 data : d
41161             });
41162             this.valueField = 'value';
41163             this.displayField = 'text';
41164         }
41165         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41166         if(!this.lazyRender){
41167             this.target = true;
41168             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41169             s.parentNode.removeChild(s); // remove it
41170             this.render(this.el.parentNode);
41171         }else{
41172             s.parentNode.removeChild(s); // remove it
41173         }
41174
41175     }
41176     if (this.store) {
41177         this.store = Roo.factory(this.store, Roo.data);
41178     }
41179     
41180     this.selectedIndex = -1;
41181     if(this.mode == 'local'){
41182         if(config.queryDelay === undefined){
41183             this.queryDelay = 10;
41184         }
41185         if(config.minChars === undefined){
41186             this.minChars = 0;
41187         }
41188     }
41189 };
41190
41191 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41192     /**
41193      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41194      */
41195     /**
41196      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41197      * rendering into an Roo.Editor, defaults to false)
41198      */
41199     /**
41200      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41201      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41202      */
41203     /**
41204      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41205      */
41206     /**
41207      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41208      * the dropdown list (defaults to undefined, with no header element)
41209      */
41210
41211      /**
41212      * @cfg {String/Roo.Template} tpl The template to use to render the output
41213      */
41214      
41215     // private
41216     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41217     /**
41218      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41219      */
41220     listWidth: undefined,
41221     /**
41222      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41223      * mode = 'remote' or 'text' if mode = 'local')
41224      */
41225     displayField: undefined,
41226     /**
41227      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41228      * mode = 'remote' or 'value' if mode = 'local'). 
41229      * Note: use of a valueField requires the user make a selection
41230      * in order for a value to be mapped.
41231      */
41232     valueField: undefined,
41233     
41234     
41235     /**
41236      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41237      * field's data value (defaults to the underlying DOM element's name)
41238      */
41239     hiddenName: undefined,
41240     /**
41241      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41242      */
41243     listClass: '',
41244     /**
41245      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41246      */
41247     selectedClass: 'x-combo-selected',
41248     /**
41249      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41250      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41251      * which displays a downward arrow icon).
41252      */
41253     triggerClass : 'x-form-arrow-trigger',
41254     /**
41255      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41256      */
41257     shadow:'sides',
41258     /**
41259      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41260      * anchor positions (defaults to 'tl-bl')
41261      */
41262     listAlign: 'tl-bl?',
41263     /**
41264      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41265      */
41266     maxHeight: 300,
41267     /**
41268      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41269      * query specified by the allQuery config option (defaults to 'query')
41270      */
41271     triggerAction: 'query',
41272     /**
41273      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41274      * (defaults to 4, does not apply if editable = false)
41275      */
41276     minChars : 4,
41277     /**
41278      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41279      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41280      */
41281     typeAhead: false,
41282     /**
41283      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41284      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41285      */
41286     queryDelay: 500,
41287     /**
41288      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41289      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41290      */
41291     pageSize: 0,
41292     /**
41293      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41294      * when editable = true (defaults to false)
41295      */
41296     selectOnFocus:false,
41297     /**
41298      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41299      */
41300     queryParam: 'query',
41301     /**
41302      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41303      * when mode = 'remote' (defaults to 'Loading...')
41304      */
41305     loadingText: 'Loading...',
41306     /**
41307      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41308      */
41309     resizable: false,
41310     /**
41311      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41312      */
41313     handleHeight : 8,
41314     /**
41315      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41316      * traditional select (defaults to true)
41317      */
41318     editable: true,
41319     /**
41320      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41321      */
41322     allQuery: '',
41323     /**
41324      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41325      */
41326     mode: 'remote',
41327     /**
41328      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41329      * listWidth has a higher value)
41330      */
41331     minListWidth : 70,
41332     /**
41333      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41334      * allow the user to set arbitrary text into the field (defaults to false)
41335      */
41336     forceSelection:false,
41337     /**
41338      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41339      * if typeAhead = true (defaults to 250)
41340      */
41341     typeAheadDelay : 250,
41342     /**
41343      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41344      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41345      */
41346     valueNotFoundText : undefined,
41347     /**
41348      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41349      */
41350     blockFocus : false,
41351     
41352     /**
41353      * @cfg {Boolean} disableClear Disable showing of clear button.
41354      */
41355     disableClear : false,
41356     /**
41357      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41358      */
41359     alwaysQuery : false,
41360     
41361     //private
41362     addicon : false,
41363     editicon: false,
41364     
41365     // element that contains real text value.. (when hidden is used..)
41366      
41367     // private
41368     onRender : function(ct, position){
41369         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41370         if(this.hiddenName){
41371             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41372                     'before', true);
41373             this.hiddenField.value =
41374                 this.hiddenValue !== undefined ? this.hiddenValue :
41375                 this.value !== undefined ? this.value : '';
41376
41377             // prevent input submission
41378             this.el.dom.removeAttribute('name');
41379              
41380              
41381         }
41382         if(Roo.isGecko){
41383             this.el.dom.setAttribute('autocomplete', 'off');
41384         }
41385
41386         var cls = 'x-combo-list';
41387
41388         this.list = new Roo.Layer({
41389             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41390         });
41391
41392         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41393         this.list.setWidth(lw);
41394         this.list.swallowEvent('mousewheel');
41395         this.assetHeight = 0;
41396
41397         if(this.title){
41398             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41399             this.assetHeight += this.header.getHeight();
41400         }
41401
41402         this.innerList = this.list.createChild({cls:cls+'-inner'});
41403         this.innerList.on('mouseover', this.onViewOver, this);
41404         this.innerList.on('mousemove', this.onViewMove, this);
41405         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41406         
41407         if(this.allowBlank && !this.pageSize && !this.disableClear){
41408             this.footer = this.list.createChild({cls:cls+'-ft'});
41409             this.pageTb = new Roo.Toolbar(this.footer);
41410            
41411         }
41412         if(this.pageSize){
41413             this.footer = this.list.createChild({cls:cls+'-ft'});
41414             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41415                     {pageSize: this.pageSize});
41416             
41417         }
41418         
41419         if (this.pageTb && this.allowBlank && !this.disableClear) {
41420             var _this = this;
41421             this.pageTb.add(new Roo.Toolbar.Fill(), {
41422                 cls: 'x-btn-icon x-btn-clear',
41423                 text: '&#160;',
41424                 handler: function()
41425                 {
41426                     _this.collapse();
41427                     _this.clearValue();
41428                     _this.onSelect(false, -1);
41429                 }
41430             });
41431         }
41432         if (this.footer) {
41433             this.assetHeight += this.footer.getHeight();
41434         }
41435         
41436
41437         if(!this.tpl){
41438             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41439         }
41440
41441         this.view = new Roo.View(this.innerList, this.tpl, {
41442             singleSelect:true, store: this.store, selectedClass: this.selectedClass
41443         });
41444
41445         this.view.on('click', this.onViewClick, this);
41446
41447         this.store.on('beforeload', this.onBeforeLoad, this);
41448         this.store.on('load', this.onLoad, this);
41449         this.store.on('loadexception', this.onLoadException, this);
41450
41451         if(this.resizable){
41452             this.resizer = new Roo.Resizable(this.list,  {
41453                pinned:true, handles:'se'
41454             });
41455             this.resizer.on('resize', function(r, w, h){
41456                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41457                 this.listWidth = w;
41458                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41459                 this.restrictHeight();
41460             }, this);
41461             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41462         }
41463         if(!this.editable){
41464             this.editable = true;
41465             this.setEditable(false);
41466         }  
41467         
41468         
41469         if (typeof(this.events.add.listeners) != 'undefined') {
41470             
41471             this.addicon = this.wrap.createChild(
41472                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41473        
41474             this.addicon.on('click', function(e) {
41475                 this.fireEvent('add', this);
41476             }, this);
41477         }
41478         if (typeof(this.events.edit.listeners) != 'undefined') {
41479             
41480             this.editicon = this.wrap.createChild(
41481                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41482             if (this.addicon) {
41483                 this.editicon.setStyle('margin-left', '40px');
41484             }
41485             this.editicon.on('click', function(e) {
41486                 
41487                 // we fire even  if inothing is selected..
41488                 this.fireEvent('edit', this, this.lastData );
41489                 
41490             }, this);
41491         }
41492         
41493         
41494         
41495     },
41496
41497     // private
41498     initEvents : function(){
41499         Roo.form.ComboBox.superclass.initEvents.call(this);
41500
41501         this.keyNav = new Roo.KeyNav(this.el, {
41502             "up" : function(e){
41503                 this.inKeyMode = true;
41504                 this.selectPrev();
41505             },
41506
41507             "down" : function(e){
41508                 if(!this.isExpanded()){
41509                     this.onTriggerClick();
41510                 }else{
41511                     this.inKeyMode = true;
41512                     this.selectNext();
41513                 }
41514             },
41515
41516             "enter" : function(e){
41517                 this.onViewClick();
41518                 //return true;
41519             },
41520
41521             "esc" : function(e){
41522                 this.collapse();
41523             },
41524
41525             "tab" : function(e){
41526                 this.onViewClick(false);
41527                 this.fireEvent("specialkey", this, e);
41528                 return true;
41529             },
41530
41531             scope : this,
41532
41533             doRelay : function(foo, bar, hname){
41534                 if(hname == 'down' || this.scope.isExpanded()){
41535                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41536                 }
41537                 return true;
41538             },
41539
41540             forceKeyDown: true
41541         });
41542         this.queryDelay = Math.max(this.queryDelay || 10,
41543                 this.mode == 'local' ? 10 : 250);
41544         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41545         if(this.typeAhead){
41546             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41547         }
41548         if(this.editable !== false){
41549             this.el.on("keyup", this.onKeyUp, this);
41550         }
41551         if(this.forceSelection){
41552             this.on('blur', this.doForce, this);
41553         }
41554     },
41555
41556     onDestroy : function(){
41557         if(this.view){
41558             this.view.setStore(null);
41559             this.view.el.removeAllListeners();
41560             this.view.el.remove();
41561             this.view.purgeListeners();
41562         }
41563         if(this.list){
41564             this.list.destroy();
41565         }
41566         if(this.store){
41567             this.store.un('beforeload', this.onBeforeLoad, this);
41568             this.store.un('load', this.onLoad, this);
41569             this.store.un('loadexception', this.onLoadException, this);
41570         }
41571         Roo.form.ComboBox.superclass.onDestroy.call(this);
41572     },
41573
41574     // private
41575     fireKey : function(e){
41576         if(e.isNavKeyPress() && !this.list.isVisible()){
41577             this.fireEvent("specialkey", this, e);
41578         }
41579     },
41580
41581     // private
41582     onResize: function(w, h){
41583         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41584         
41585         if(typeof w != 'number'){
41586             // we do not handle it!?!?
41587             return;
41588         }
41589         var tw = this.trigger.getWidth();
41590         tw += this.addicon ? this.addicon.getWidth() : 0;
41591         tw += this.editicon ? this.editicon.getWidth() : 0;
41592         var x = w - tw;
41593         this.el.setWidth( this.adjustWidth('input', x));
41594             
41595         this.trigger.setStyle('left', x+'px');
41596         
41597         if(this.list && this.listWidth === undefined){
41598             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41599             this.list.setWidth(lw);
41600             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41601         }
41602         
41603     
41604         
41605     },
41606
41607     /**
41608      * Allow or prevent the user from directly editing the field text.  If false is passed,
41609      * the user will only be able to select from the items defined in the dropdown list.  This method
41610      * is the runtime equivalent of setting the 'editable' config option at config time.
41611      * @param {Boolean} value True to allow the user to directly edit the field text
41612      */
41613     setEditable : function(value){
41614         if(value == this.editable){
41615             return;
41616         }
41617         this.editable = value;
41618         if(!value){
41619             this.el.dom.setAttribute('readOnly', true);
41620             this.el.on('mousedown', this.onTriggerClick,  this);
41621             this.el.addClass('x-combo-noedit');
41622         }else{
41623             this.el.dom.setAttribute('readOnly', false);
41624             this.el.un('mousedown', this.onTriggerClick,  this);
41625             this.el.removeClass('x-combo-noedit');
41626         }
41627     },
41628
41629     // private
41630     onBeforeLoad : function(){
41631         if(!this.hasFocus){
41632             return;
41633         }
41634         this.innerList.update(this.loadingText ?
41635                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41636         this.restrictHeight();
41637         this.selectedIndex = -1;
41638     },
41639
41640     // private
41641     onLoad : function(){
41642         if(!this.hasFocus){
41643             return;
41644         }
41645         if(this.store.getCount() > 0){
41646             this.expand();
41647             this.restrictHeight();
41648             if(this.lastQuery == this.allQuery){
41649                 if(this.editable){
41650                     this.el.dom.select();
41651                 }
41652                 if(!this.selectByValue(this.value, true)){
41653                     this.select(0, true);
41654                 }
41655             }else{
41656                 this.selectNext();
41657                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41658                     this.taTask.delay(this.typeAheadDelay);
41659                 }
41660             }
41661         }else{
41662             this.onEmptyResults();
41663         }
41664         //this.el.focus();
41665     },
41666     // private
41667     onLoadException : function()
41668     {
41669         this.collapse();
41670         Roo.log(this.store.reader.jsonData);
41671         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41672             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41673         }
41674         
41675         
41676     },
41677     // private
41678     onTypeAhead : function(){
41679         if(this.store.getCount() > 0){
41680             var r = this.store.getAt(0);
41681             var newValue = r.data[this.displayField];
41682             var len = newValue.length;
41683             var selStart = this.getRawValue().length;
41684             if(selStart != len){
41685                 this.setRawValue(newValue);
41686                 this.selectText(selStart, newValue.length);
41687             }
41688         }
41689     },
41690
41691     // private
41692     onSelect : function(record, index){
41693         if(this.fireEvent('beforeselect', this, record, index) !== false){
41694             this.setFromData(index > -1 ? record.data : false);
41695             this.collapse();
41696             this.fireEvent('select', this, record, index);
41697         }
41698     },
41699
41700     /**
41701      * Returns the currently selected field value or empty string if no value is set.
41702      * @return {String} value The selected value
41703      */
41704     getValue : function(){
41705         if(this.valueField){
41706             return typeof this.value != 'undefined' ? this.value : '';
41707         }
41708         return Roo.form.ComboBox.superclass.getValue.call(this);
41709     },
41710
41711     /**
41712      * Clears any text/value currently set in the field
41713      */
41714     clearValue : function(){
41715         if(this.hiddenField){
41716             this.hiddenField.value = '';
41717         }
41718         this.value = '';
41719         this.setRawValue('');
41720         this.lastSelectionText = '';
41721         
41722     },
41723
41724     /**
41725      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41726      * will be displayed in the field.  If the value does not match the data value of an existing item,
41727      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41728      * Otherwise the field will be blank (although the value will still be set).
41729      * @param {String} value The value to match
41730      */
41731     setValue : function(v){
41732         var text = v;
41733         if(this.valueField){
41734             var r = this.findRecord(this.valueField, v);
41735             if(r){
41736                 text = r.data[this.displayField];
41737             }else if(this.valueNotFoundText !== undefined){
41738                 text = this.valueNotFoundText;
41739             }
41740         }
41741         this.lastSelectionText = text;
41742         if(this.hiddenField){
41743             this.hiddenField.value = v;
41744         }
41745         Roo.form.ComboBox.superclass.setValue.call(this, text);
41746         this.value = v;
41747     },
41748     /**
41749      * @property {Object} the last set data for the element
41750      */
41751     
41752     lastData : false,
41753     /**
41754      * Sets the value of the field based on a object which is related to the record format for the store.
41755      * @param {Object} value the value to set as. or false on reset?
41756      */
41757     setFromData : function(o){
41758         var dv = ''; // display value
41759         var vv = ''; // value value..
41760         this.lastData = o;
41761         if (this.displayField) {
41762             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41763         } else {
41764             // this is an error condition!!!
41765             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41766         }
41767         
41768         if(this.valueField){
41769             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41770         }
41771         if(this.hiddenField){
41772             this.hiddenField.value = vv;
41773             
41774             this.lastSelectionText = dv;
41775             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41776             this.value = vv;
41777             return;
41778         }
41779         // no hidden field.. - we store the value in 'value', but still display
41780         // display field!!!!
41781         this.lastSelectionText = dv;
41782         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41783         this.value = vv;
41784         
41785         
41786     },
41787     // private
41788     reset : function(){
41789         // overridden so that last data is reset..
41790         this.setValue(this.resetValue);
41791         this.originalValue = this.getValue();
41792         this.clearInvalid();
41793         this.lastData = false;
41794         if (this.view) {
41795             this.view.clearSelections();
41796         }
41797     },
41798     // private
41799     findRecord : function(prop, value){
41800         var record;
41801         if(this.store.getCount() > 0){
41802             this.store.each(function(r){
41803                 if(r.data[prop] == value){
41804                     record = r;
41805                     return false;
41806                 }
41807                 return true;
41808             });
41809         }
41810         return record;
41811     },
41812     
41813     getName: function()
41814     {
41815         // returns hidden if it's set..
41816         if (!this.rendered) {return ''};
41817         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41818         
41819     },
41820     // private
41821     onViewMove : function(e, t){
41822         this.inKeyMode = false;
41823     },
41824
41825     // private
41826     onViewOver : function(e, t){
41827         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41828             return;
41829         }
41830         var item = this.view.findItemFromChild(t);
41831         if(item){
41832             var index = this.view.indexOf(item);
41833             this.select(index, false);
41834         }
41835     },
41836
41837     // private
41838     onViewClick : function(doFocus)
41839     {
41840         var index = this.view.getSelectedIndexes()[0];
41841         var r = this.store.getAt(index);
41842         if(r){
41843             this.onSelect(r, index);
41844         }
41845         if(doFocus !== false && !this.blockFocus){
41846             this.el.focus();
41847         }
41848     },
41849
41850     // private
41851     restrictHeight : function(){
41852         this.innerList.dom.style.height = '';
41853         var inner = this.innerList.dom;
41854         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41855         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41856         this.list.beginUpdate();
41857         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41858         this.list.alignTo(this.el, this.listAlign);
41859         this.list.endUpdate();
41860     },
41861
41862     // private
41863     onEmptyResults : function(){
41864         this.collapse();
41865     },
41866
41867     /**
41868      * Returns true if the dropdown list is expanded, else false.
41869      */
41870     isExpanded : function(){
41871         return this.list.isVisible();
41872     },
41873
41874     /**
41875      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41876      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41877      * @param {String} value The data value of the item to select
41878      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41879      * selected item if it is not currently in view (defaults to true)
41880      * @return {Boolean} True if the value matched an item in the list, else false
41881      */
41882     selectByValue : function(v, scrollIntoView){
41883         if(v !== undefined && v !== null){
41884             var r = this.findRecord(this.valueField || this.displayField, v);
41885             if(r){
41886                 this.select(this.store.indexOf(r), scrollIntoView);
41887                 return true;
41888             }
41889         }
41890         return false;
41891     },
41892
41893     /**
41894      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41895      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41896      * @param {Number} index The zero-based index of the list item to select
41897      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41898      * selected item if it is not currently in view (defaults to true)
41899      */
41900     select : function(index, scrollIntoView){
41901         this.selectedIndex = index;
41902         this.view.select(index);
41903         if(scrollIntoView !== false){
41904             var el = this.view.getNode(index);
41905             if(el){
41906                 this.innerList.scrollChildIntoView(el, false);
41907             }
41908         }
41909     },
41910
41911     // private
41912     selectNext : function(){
41913         var ct = this.store.getCount();
41914         if(ct > 0){
41915             if(this.selectedIndex == -1){
41916                 this.select(0);
41917             }else if(this.selectedIndex < ct-1){
41918                 this.select(this.selectedIndex+1);
41919             }
41920         }
41921     },
41922
41923     // private
41924     selectPrev : function(){
41925         var ct = this.store.getCount();
41926         if(ct > 0){
41927             if(this.selectedIndex == -1){
41928                 this.select(0);
41929             }else if(this.selectedIndex != 0){
41930                 this.select(this.selectedIndex-1);
41931             }
41932         }
41933     },
41934
41935     // private
41936     onKeyUp : function(e){
41937         if(this.editable !== false && !e.isSpecialKey()){
41938             this.lastKey = e.getKey();
41939             this.dqTask.delay(this.queryDelay);
41940         }
41941     },
41942
41943     // private
41944     validateBlur : function(){
41945         return !this.list || !this.list.isVisible();   
41946     },
41947
41948     // private
41949     initQuery : function(){
41950         this.doQuery(this.getRawValue());
41951     },
41952
41953     // private
41954     doForce : function(){
41955         if(this.el.dom.value.length > 0){
41956             this.el.dom.value =
41957                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
41958              
41959         }
41960     },
41961
41962     /**
41963      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
41964      * query allowing the query action to be canceled if needed.
41965      * @param {String} query The SQL query to execute
41966      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
41967      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
41968      * saved in the current store (defaults to false)
41969      */
41970     doQuery : function(q, forceAll){
41971         if(q === undefined || q === null){
41972             q = '';
41973         }
41974         var qe = {
41975             query: q,
41976             forceAll: forceAll,
41977             combo: this,
41978             cancel:false
41979         };
41980         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
41981             return false;
41982         }
41983         q = qe.query;
41984         forceAll = qe.forceAll;
41985         if(forceAll === true || (q.length >= this.minChars)){
41986             if(this.lastQuery != q || this.alwaysQuery){
41987                 this.lastQuery = q;
41988                 if(this.mode == 'local'){
41989                     this.selectedIndex = -1;
41990                     if(forceAll){
41991                         this.store.clearFilter();
41992                     }else{
41993                         this.store.filter(this.displayField, q);
41994                     }
41995                     this.onLoad();
41996                 }else{
41997                     this.store.baseParams[this.queryParam] = q;
41998                     this.store.load({
41999                         params: this.getParams(q)
42000                     });
42001                     this.expand();
42002                 }
42003             }else{
42004                 this.selectedIndex = -1;
42005                 this.onLoad();   
42006             }
42007         }
42008     },
42009
42010     // private
42011     getParams : function(q){
42012         var p = {};
42013         //p[this.queryParam] = q;
42014         if(this.pageSize){
42015             p.start = 0;
42016             p.limit = this.pageSize;
42017         }
42018         return p;
42019     },
42020
42021     /**
42022      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42023      */
42024     collapse : function(){
42025         if(!this.isExpanded()){
42026             return;
42027         }
42028         this.list.hide();
42029         Roo.get(document).un('mousedown', this.collapseIf, this);
42030         Roo.get(document).un('mousewheel', this.collapseIf, this);
42031         if (!this.editable) {
42032             Roo.get(document).un('keydown', this.listKeyPress, this);
42033         }
42034         this.fireEvent('collapse', this);
42035     },
42036
42037     // private
42038     collapseIf : function(e){
42039         if(!e.within(this.wrap) && !e.within(this.list)){
42040             this.collapse();
42041         }
42042     },
42043
42044     /**
42045      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42046      */
42047     expand : function(){
42048         if(this.isExpanded() || !this.hasFocus){
42049             return;
42050         }
42051         this.list.alignTo(this.el, this.listAlign);
42052         this.list.show();
42053         Roo.get(document).on('mousedown', this.collapseIf, this);
42054         Roo.get(document).on('mousewheel', this.collapseIf, this);
42055         if (!this.editable) {
42056             Roo.get(document).on('keydown', this.listKeyPress, this);
42057         }
42058         
42059         this.fireEvent('expand', this);
42060     },
42061
42062     // private
42063     // Implements the default empty TriggerField.onTriggerClick function
42064     onTriggerClick : function(){
42065         if(this.disabled){
42066             return;
42067         }
42068         if(this.isExpanded()){
42069             this.collapse();
42070             if (!this.blockFocus) {
42071                 this.el.focus();
42072             }
42073             
42074         }else {
42075             this.hasFocus = true;
42076             if(this.triggerAction == 'all') {
42077                 this.doQuery(this.allQuery, true);
42078             } else {
42079                 this.doQuery(this.getRawValue());
42080             }
42081             if (!this.blockFocus) {
42082                 this.el.focus();
42083             }
42084         }
42085     },
42086     listKeyPress : function(e)
42087     {
42088         //Roo.log('listkeypress');
42089         // scroll to first matching element based on key pres..
42090         if (e.isSpecialKey()) {
42091             return false;
42092         }
42093         var k = String.fromCharCode(e.getKey()).toUpperCase();
42094         //Roo.log(k);
42095         var match  = false;
42096         var csel = this.view.getSelectedNodes();
42097         var cselitem = false;
42098         if (csel.length) {
42099             var ix = this.view.indexOf(csel[0]);
42100             cselitem  = this.store.getAt(ix);
42101             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42102                 cselitem = false;
42103             }
42104             
42105         }
42106         
42107         this.store.each(function(v) { 
42108             if (cselitem) {
42109                 // start at existing selection.
42110                 if (cselitem.id == v.id) {
42111                     cselitem = false;
42112                 }
42113                 return;
42114             }
42115                 
42116             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42117                 match = this.store.indexOf(v);
42118                 return false;
42119             }
42120         }, this);
42121         
42122         if (match === false) {
42123             return true; // no more action?
42124         }
42125         // scroll to?
42126         this.view.select(match);
42127         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42128         sn.scrollIntoView(sn.dom.parentNode, false);
42129     } 
42130
42131     /** 
42132     * @cfg {Boolean} grow 
42133     * @hide 
42134     */
42135     /** 
42136     * @cfg {Number} growMin 
42137     * @hide 
42138     */
42139     /** 
42140     * @cfg {Number} growMax 
42141     * @hide 
42142     */
42143     /**
42144      * @hide
42145      * @method autoSize
42146      */
42147 });/*
42148  * Copyright(c) 2010-2012, Roo J Solutions Limited
42149  *
42150  * Licence LGPL
42151  *
42152  */
42153
42154 /**
42155  * @class Roo.form.ComboBoxArray
42156  * @extends Roo.form.TextField
42157  * A facebook style adder... for lists of email / people / countries  etc...
42158  * pick multiple items from a combo box, and shows each one.
42159  *
42160  *  Fred [x]  Brian [x]  [Pick another |v]
42161  *
42162  *
42163  *  For this to work: it needs various extra information
42164  *    - normal combo problay has
42165  *      name, hiddenName
42166  *    + displayField, valueField
42167  *
42168  *    For our purpose...
42169  *
42170  *
42171  *   If we change from 'extends' to wrapping...
42172  *   
42173  *  
42174  *
42175  
42176  
42177  * @constructor
42178  * Create a new ComboBoxArray.
42179  * @param {Object} config Configuration options
42180  */
42181  
42182
42183 Roo.form.ComboBoxArray = function(config)
42184 {
42185     this.addEvents({
42186         /**
42187          * @event beforeremove
42188          * Fires before remove the value from the list
42189              * @param {Roo.form.ComboBoxArray} _self This combo box array
42190              * @param {Roo.form.ComboBoxArray.Item} item removed item
42191              */
42192         'beforeremove' : true,
42193         /**
42194          * @event remove
42195          * Fires when remove the value from the list
42196              * @param {Roo.form.ComboBoxArray} _self This combo box array
42197              * @param {Roo.form.ComboBoxArray.Item} item removed item
42198              */
42199         'remove' : true
42200         
42201         
42202     });
42203     
42204     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42205     
42206     this.items = new Roo.util.MixedCollection(false);
42207     
42208     // construct the child combo...
42209     
42210     
42211     
42212     
42213    
42214     
42215 }
42216
42217  
42218 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42219
42220     /**
42221      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42222      */
42223     
42224     lastData : false,
42225     
42226     // behavies liek a hiddne field
42227     inputType:      'hidden',
42228     /**
42229      * @cfg {Number} width The width of the box that displays the selected element
42230      */ 
42231     width:          300,
42232
42233     
42234     
42235     /**
42236      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42237      */
42238     name : false,
42239     /**
42240      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42241      */
42242     hiddenName : false,
42243     
42244     
42245     // private the array of items that are displayed..
42246     items  : false,
42247     // private - the hidden field el.
42248     hiddenEl : false,
42249     // private - the filed el..
42250     el : false,
42251     
42252     //validateValue : function() { return true; }, // all values are ok!
42253     //onAddClick: function() { },
42254     
42255     onRender : function(ct, position) 
42256     {
42257         
42258         // create the standard hidden element
42259         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42260         
42261         
42262         // give fake names to child combo;
42263         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42264         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42265         
42266         this.combo = Roo.factory(this.combo, Roo.form);
42267         this.combo.onRender(ct, position);
42268         if (typeof(this.combo.width) != 'undefined') {
42269             this.combo.onResize(this.combo.width,0);
42270         }
42271         
42272         this.combo.initEvents();
42273         
42274         // assigned so form know we need to do this..
42275         this.store          = this.combo.store;
42276         this.valueField     = this.combo.valueField;
42277         this.displayField   = this.combo.displayField ;
42278         
42279         
42280         this.combo.wrap.addClass('x-cbarray-grp');
42281         
42282         var cbwrap = this.combo.wrap.createChild(
42283             {tag: 'div', cls: 'x-cbarray-cb'},
42284             this.combo.el.dom
42285         );
42286         
42287              
42288         this.hiddenEl = this.combo.wrap.createChild({
42289             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42290         });
42291         this.el = this.combo.wrap.createChild({
42292             tag: 'input',  type:'hidden' , name: this.name, value : ''
42293         });
42294          //   this.el.dom.removeAttribute("name");
42295         
42296         
42297         this.outerWrap = this.combo.wrap;
42298         this.wrap = cbwrap;
42299         
42300         this.outerWrap.setWidth(this.width);
42301         this.outerWrap.dom.removeChild(this.el.dom);
42302         
42303         this.wrap.dom.appendChild(this.el.dom);
42304         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42305         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42306         
42307         this.combo.trigger.setStyle('position','relative');
42308         this.combo.trigger.setStyle('left', '0px');
42309         this.combo.trigger.setStyle('top', '2px');
42310         
42311         this.combo.el.setStyle('vertical-align', 'text-bottom');
42312         
42313         //this.trigger.setStyle('vertical-align', 'top');
42314         
42315         // this should use the code from combo really... on('add' ....)
42316         if (this.adder) {
42317             
42318         
42319             this.adder = this.outerWrap.createChild(
42320                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42321             var _t = this;
42322             this.adder.on('click', function(e) {
42323                 _t.fireEvent('adderclick', this, e);
42324             }, _t);
42325         }
42326         //var _t = this;
42327         //this.adder.on('click', this.onAddClick, _t);
42328         
42329         
42330         this.combo.on('select', function(cb, rec, ix) {
42331             this.addItem(rec.data);
42332             
42333             cb.setValue('');
42334             cb.el.dom.value = '';
42335             //cb.lastData = rec.data;
42336             // add to list
42337             
42338         }, this);
42339         
42340         
42341     },
42342     
42343     
42344     getName: function()
42345     {
42346         // returns hidden if it's set..
42347         if (!this.rendered) {return ''};
42348         return  this.hiddenName ? this.hiddenName : this.name;
42349         
42350     },
42351     
42352     
42353     onResize: function(w, h){
42354         
42355         return;
42356         // not sure if this is needed..
42357         //this.combo.onResize(w,h);
42358         
42359         if(typeof w != 'number'){
42360             // we do not handle it!?!?
42361             return;
42362         }
42363         var tw = this.combo.trigger.getWidth();
42364         tw += this.addicon ? this.addicon.getWidth() : 0;
42365         tw += this.editicon ? this.editicon.getWidth() : 0;
42366         var x = w - tw;
42367         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42368             
42369         this.combo.trigger.setStyle('left', '0px');
42370         
42371         if(this.list && this.listWidth === undefined){
42372             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42373             this.list.setWidth(lw);
42374             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42375         }
42376         
42377     
42378         
42379     },
42380     
42381     addItem: function(rec)
42382     {
42383         var valueField = this.combo.valueField;
42384         var displayField = this.combo.displayField;
42385         
42386         if (this.items.indexOfKey(rec[valueField]) > -1) {
42387             //console.log("GOT " + rec.data.id);
42388             return;
42389         }
42390         
42391         var x = new Roo.form.ComboBoxArray.Item({
42392             //id : rec[this.idField],
42393             data : rec,
42394             displayField : displayField ,
42395             tipField : displayField ,
42396             cb : this
42397         });
42398         // use the 
42399         this.items.add(rec[valueField],x);
42400         // add it before the element..
42401         this.updateHiddenEl();
42402         x.render(this.outerWrap, this.wrap.dom);
42403         // add the image handler..
42404     },
42405     
42406     updateHiddenEl : function()
42407     {
42408         this.validate();
42409         if (!this.hiddenEl) {
42410             return;
42411         }
42412         var ar = [];
42413         var idField = this.combo.valueField;
42414         
42415         this.items.each(function(f) {
42416             ar.push(f.data[idField]);
42417         });
42418         this.hiddenEl.dom.value = ar.join(',');
42419         this.validate();
42420     },
42421     
42422     reset : function()
42423     {
42424         this.items.clear();
42425         
42426         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42427            el.remove();
42428         });
42429         
42430         this.el.dom.value = '';
42431         if (this.hiddenEl) {
42432             this.hiddenEl.dom.value = '';
42433         }
42434         
42435     },
42436     getValue: function()
42437     {
42438         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42439     },
42440     setValue: function(v) // not a valid action - must use addItems..
42441     {
42442         
42443         this.reset();
42444          
42445         if (this.store.isLocal && (typeof(v) == 'string')) {
42446             // then we can use the store to find the values..
42447             // comma seperated at present.. this needs to allow JSON based encoding..
42448             this.hiddenEl.value  = v;
42449             var v_ar = [];
42450             Roo.each(v.split(','), function(k) {
42451                 Roo.log("CHECK " + this.valueField + ',' + k);
42452                 var li = this.store.query(this.valueField, k);
42453                 if (!li.length) {
42454                     return;
42455                 }
42456                 var add = {};
42457                 add[this.valueField] = k;
42458                 add[this.displayField] = li.item(0).data[this.displayField];
42459                 
42460                 this.addItem(add);
42461             }, this) 
42462              
42463         }
42464         if (typeof(v) == 'object' ) {
42465             // then let's assume it's an array of objects..
42466             Roo.each(v, function(l) {
42467                 this.addItem(l);
42468             }, this);
42469              
42470         }
42471         
42472         
42473     },
42474     setFromData: function(v)
42475     {
42476         // this recieves an object, if setValues is called.
42477         this.reset();
42478         this.el.dom.value = v[this.displayField];
42479         this.hiddenEl.dom.value = v[this.valueField];
42480         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42481             return;
42482         }
42483         var kv = v[this.valueField];
42484         var dv = v[this.displayField];
42485         kv = typeof(kv) != 'string' ? '' : kv;
42486         dv = typeof(dv) != 'string' ? '' : dv;
42487         
42488         
42489         var keys = kv.split(',');
42490         var display = dv.split(',');
42491         for (var i = 0 ; i < keys.length; i++) {
42492             
42493             add = {};
42494             add[this.valueField] = keys[i];
42495             add[this.displayField] = display[i];
42496             this.addItem(add);
42497         }
42498       
42499         
42500     },
42501     
42502     /**
42503      * Validates the combox array value
42504      * @return {Boolean} True if the value is valid, else false
42505      */
42506     validate : function(){
42507         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42508             this.clearInvalid();
42509             return true;
42510         }
42511         return false;
42512     },
42513     
42514     validateValue : function(value){
42515         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42516         
42517     },
42518     
42519     /*@
42520      * overide
42521      * 
42522      */
42523     isDirty : function() {
42524         if(this.disabled) {
42525             return false;
42526         }
42527         
42528         try {
42529             var d = Roo.decode(String(this.originalValue));
42530         } catch (e) {
42531             return String(this.getValue()) !== String(this.originalValue);
42532         }
42533         
42534         var originalValue = [];
42535         
42536         for (var i = 0; i < d.length; i++){
42537             originalValue.push(d[i][this.valueField]);
42538         }
42539         
42540         return String(this.getValue()) !== String(originalValue.join(','));
42541         
42542     }
42543     
42544 });
42545
42546
42547
42548 /**
42549  * @class Roo.form.ComboBoxArray.Item
42550  * @extends Roo.BoxComponent
42551  * A selected item in the list
42552  *  Fred [x]  Brian [x]  [Pick another |v]
42553  * 
42554  * @constructor
42555  * Create a new item.
42556  * @param {Object} config Configuration options
42557  */
42558  
42559 Roo.form.ComboBoxArray.Item = function(config) {
42560     config.id = Roo.id();
42561     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42562 }
42563
42564 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42565     data : {},
42566     cb: false,
42567     displayField : false,
42568     tipField : false,
42569     
42570     
42571     defaultAutoCreate : {
42572         tag: 'div',
42573         cls: 'x-cbarray-item',
42574         cn : [ 
42575             { tag: 'div' },
42576             {
42577                 tag: 'img',
42578                 width:16,
42579                 height : 16,
42580                 src : Roo.BLANK_IMAGE_URL ,
42581                 align: 'center'
42582             }
42583         ]
42584         
42585     },
42586     
42587  
42588     onRender : function(ct, position)
42589     {
42590         Roo.form.Field.superclass.onRender.call(this, ct, position);
42591         
42592         if(!this.el){
42593             var cfg = this.getAutoCreate();
42594             this.el = ct.createChild(cfg, position);
42595         }
42596         
42597         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42598         
42599         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42600             this.cb.renderer(this.data) :
42601             String.format('{0}',this.data[this.displayField]);
42602         
42603             
42604         this.el.child('div').dom.setAttribute('qtip',
42605                         String.format('{0}',this.data[this.tipField])
42606         );
42607         
42608         this.el.child('img').on('click', this.remove, this);
42609         
42610     },
42611    
42612     remove : function()
42613     {
42614         if(this.cb.disabled){
42615             return;
42616         }
42617         
42618         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42619             this.cb.items.remove(this);
42620             this.el.child('img').un('click', this.remove, this);
42621             this.el.remove();
42622             this.cb.updateHiddenEl();
42623
42624             this.cb.fireEvent('remove', this.cb, this);
42625         }
42626         
42627     }
42628 });/*
42629  * Based on:
42630  * Ext JS Library 1.1.1
42631  * Copyright(c) 2006-2007, Ext JS, LLC.
42632  *
42633  * Originally Released Under LGPL - original licence link has changed is not relivant.
42634  *
42635  * Fork - LGPL
42636  * <script type="text/javascript">
42637  */
42638 /**
42639  * @class Roo.form.Checkbox
42640  * @extends Roo.form.Field
42641  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
42642  * @constructor
42643  * Creates a new Checkbox
42644  * @param {Object} config Configuration options
42645  */
42646 Roo.form.Checkbox = function(config){
42647     Roo.form.Checkbox.superclass.constructor.call(this, config);
42648     this.addEvents({
42649         /**
42650          * @event check
42651          * Fires when the checkbox is checked or unchecked.
42652              * @param {Roo.form.Checkbox} this This checkbox
42653              * @param {Boolean} checked The new checked value
42654              */
42655         check : true
42656     });
42657 };
42658
42659 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
42660     /**
42661      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
42662      */
42663     focusClass : undefined,
42664     /**
42665      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
42666      */
42667     fieldClass: "x-form-field",
42668     /**
42669      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
42670      */
42671     checked: false,
42672     /**
42673      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
42674      * {tag: "input", type: "checkbox", autocomplete: "off"})
42675      */
42676     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
42677     /**
42678      * @cfg {String} boxLabel The text that appears beside the checkbox
42679      */
42680     boxLabel : "",
42681     /**
42682      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
42683      */  
42684     inputValue : '1',
42685     /**
42686      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
42687      */
42688      valueOff: '0', // value when not checked..
42689
42690     actionMode : 'viewEl', 
42691     //
42692     // private
42693     itemCls : 'x-menu-check-item x-form-item',
42694     groupClass : 'x-menu-group-item',
42695     inputType : 'hidden',
42696     
42697     
42698     inSetChecked: false, // check that we are not calling self...
42699     
42700     inputElement: false, // real input element?
42701     basedOn: false, // ????
42702     
42703     isFormField: true, // not sure where this is needed!!!!
42704
42705     onResize : function(){
42706         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
42707         if(!this.boxLabel){
42708             this.el.alignTo(this.wrap, 'c-c');
42709         }
42710     },
42711
42712     initEvents : function(){
42713         Roo.form.Checkbox.superclass.initEvents.call(this);
42714         this.el.on("click", this.onClick,  this);
42715         this.el.on("change", this.onClick,  this);
42716     },
42717
42718
42719     getResizeEl : function(){
42720         return this.wrap;
42721     },
42722
42723     getPositionEl : function(){
42724         return this.wrap;
42725     },
42726
42727     // private
42728     onRender : function(ct, position){
42729         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42730         /*
42731         if(this.inputValue !== undefined){
42732             this.el.dom.value = this.inputValue;
42733         }
42734         */
42735         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42736         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42737         var viewEl = this.wrap.createChild({ 
42738             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42739         this.viewEl = viewEl;   
42740         this.wrap.on('click', this.onClick,  this); 
42741         
42742         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42743         this.el.on('propertychange', this.setFromHidden,  this);  //ie
42744         
42745         
42746         
42747         if(this.boxLabel){
42748             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42749         //    viewEl.on('click', this.onClick,  this); 
42750         }
42751         //if(this.checked){
42752             this.setChecked(this.checked);
42753         //}else{
42754             //this.checked = this.el.dom;
42755         //}
42756
42757     },
42758
42759     // private
42760     initValue : Roo.emptyFn,
42761
42762     /**
42763      * Returns the checked state of the checkbox.
42764      * @return {Boolean} True if checked, else false
42765      */
42766     getValue : function(){
42767         if(this.el){
42768             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
42769         }
42770         return this.valueOff;
42771         
42772     },
42773
42774         // private
42775     onClick : function(){ 
42776         if (this.disabled) {
42777             return;
42778         }
42779         this.setChecked(!this.checked);
42780
42781         //if(this.el.dom.checked != this.checked){
42782         //    this.setValue(this.el.dom.checked);
42783        // }
42784     },
42785
42786     /**
42787      * Sets the checked state of the checkbox.
42788      * On is always based on a string comparison between inputValue and the param.
42789      * @param {Boolean/String} value - the value to set 
42790      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
42791      */
42792     setValue : function(v,suppressEvent){
42793         
42794         
42795         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
42796         //if(this.el && this.el.dom){
42797         //    this.el.dom.checked = this.checked;
42798         //    this.el.dom.defaultChecked = this.checked;
42799         //}
42800         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
42801         //this.fireEvent("check", this, this.checked);
42802     },
42803     // private..
42804     setChecked : function(state,suppressEvent)
42805     {
42806         if (this.inSetChecked) {
42807             this.checked = state;
42808             return;
42809         }
42810         
42811     
42812         if(this.wrap){
42813             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
42814         }
42815         this.checked = state;
42816         if(suppressEvent !== true){
42817             this.fireEvent('check', this, state);
42818         }
42819         this.inSetChecked = true;
42820         this.el.dom.value = state ? this.inputValue : this.valueOff;
42821         this.inSetChecked = false;
42822         
42823     },
42824     // handle setting of hidden value by some other method!!?!?
42825     setFromHidden: function()
42826     {
42827         if(!this.el){
42828             return;
42829         }
42830         //console.log("SET FROM HIDDEN");
42831         //alert('setFrom hidden');
42832         this.setValue(this.el.dom.value);
42833     },
42834     
42835     onDestroy : function()
42836     {
42837         if(this.viewEl){
42838             Roo.get(this.viewEl).remove();
42839         }
42840          
42841         Roo.form.Checkbox.superclass.onDestroy.call(this);
42842     },
42843     
42844     setBoxLabel : function(str)
42845     {
42846         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
42847     }
42848
42849 });/*
42850  * Based on:
42851  * Ext JS Library 1.1.1
42852  * Copyright(c) 2006-2007, Ext JS, LLC.
42853  *
42854  * Originally Released Under LGPL - original licence link has changed is not relivant.
42855  *
42856  * Fork - LGPL
42857  * <script type="text/javascript">
42858  */
42859  
42860 /**
42861  * @class Roo.form.Radio
42862  * @extends Roo.form.Checkbox
42863  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
42864  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
42865  * @constructor
42866  * Creates a new Radio
42867  * @param {Object} config Configuration options
42868  */
42869 Roo.form.Radio = function(){
42870     Roo.form.Radio.superclass.constructor.apply(this, arguments);
42871 };
42872 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
42873     inputType: 'radio',
42874
42875     /**
42876      * If this radio is part of a group, it will return the selected value
42877      * @return {String}
42878      */
42879     getGroupValue : function(){
42880         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
42881     },
42882     
42883     
42884     onRender : function(ct, position){
42885         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42886         
42887         if(this.inputValue !== undefined){
42888             this.el.dom.value = this.inputValue;
42889         }
42890          
42891         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42892         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42893         //var viewEl = this.wrap.createChild({ 
42894         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42895         //this.viewEl = viewEl;   
42896         //this.wrap.on('click', this.onClick,  this); 
42897         
42898         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42899         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
42900         
42901         
42902         
42903         if(this.boxLabel){
42904             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42905         //    viewEl.on('click', this.onClick,  this); 
42906         }
42907          if(this.checked){
42908             this.el.dom.checked =   'checked' ;
42909         }
42910          
42911     } 
42912     
42913     
42914 });//<script type="text/javascript">
42915
42916 /*
42917  * Based  Ext JS Library 1.1.1
42918  * Copyright(c) 2006-2007, Ext JS, LLC.
42919  * LGPL
42920  *
42921  */
42922  
42923 /**
42924  * @class Roo.HtmlEditorCore
42925  * @extends Roo.Component
42926  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
42927  *
42928  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
42929  */
42930
42931 Roo.HtmlEditorCore = function(config){
42932     
42933     
42934     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
42935     
42936     
42937     this.addEvents({
42938         /**
42939          * @event initialize
42940          * Fires when the editor is fully initialized (including the iframe)
42941          * @param {Roo.HtmlEditorCore} this
42942          */
42943         initialize: true,
42944         /**
42945          * @event activate
42946          * Fires when the editor is first receives the focus. Any insertion must wait
42947          * until after this event.
42948          * @param {Roo.HtmlEditorCore} this
42949          */
42950         activate: true,
42951          /**
42952          * @event beforesync
42953          * Fires before the textarea is updated with content from the editor iframe. Return false
42954          * to cancel the sync.
42955          * @param {Roo.HtmlEditorCore} this
42956          * @param {String} html
42957          */
42958         beforesync: true,
42959          /**
42960          * @event beforepush
42961          * Fires before the iframe editor is updated with content from the textarea. Return false
42962          * to cancel the push.
42963          * @param {Roo.HtmlEditorCore} this
42964          * @param {String} html
42965          */
42966         beforepush: true,
42967          /**
42968          * @event sync
42969          * Fires when the textarea is updated with content from the editor iframe.
42970          * @param {Roo.HtmlEditorCore} this
42971          * @param {String} html
42972          */
42973         sync: true,
42974          /**
42975          * @event push
42976          * Fires when the iframe editor is updated with content from the textarea.
42977          * @param {Roo.HtmlEditorCore} this
42978          * @param {String} html
42979          */
42980         push: true,
42981         
42982         /**
42983          * @event editorevent
42984          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
42985          * @param {Roo.HtmlEditorCore} this
42986          */
42987         editorevent: true
42988         
42989     });
42990     
42991     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
42992     
42993     // defaults : white / black...
42994     this.applyBlacklists();
42995     
42996     
42997     
42998 };
42999
43000
43001 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43002
43003
43004      /**
43005      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43006      */
43007     
43008     owner : false,
43009     
43010      /**
43011      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43012      *                        Roo.resizable.
43013      */
43014     resizable : false,
43015      /**
43016      * @cfg {Number} height (in pixels)
43017      */   
43018     height: 300,
43019    /**
43020      * @cfg {Number} width (in pixels)
43021      */   
43022     width: 500,
43023     
43024     /**
43025      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43026      * 
43027      */
43028     stylesheets: false,
43029     
43030     // id of frame..
43031     frameId: false,
43032     
43033     // private properties
43034     validationEvent : false,
43035     deferHeight: true,
43036     initialized : false,
43037     activated : false,
43038     sourceEditMode : false,
43039     onFocus : Roo.emptyFn,
43040     iframePad:3,
43041     hideMode:'offsets',
43042     
43043     clearUp: true,
43044     
43045     // blacklist + whitelisted elements..
43046     black: false,
43047     white: false,
43048      
43049     bodyCls : '',
43050
43051     /**
43052      * Protected method that will not generally be called directly. It
43053      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43054      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43055      */
43056     getDocMarkup : function(){
43057         // body styles..
43058         var st = '';
43059         
43060         // inherit styels from page...?? 
43061         if (this.stylesheets === false) {
43062             
43063             Roo.get(document.head).select('style').each(function(node) {
43064                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43065             });
43066             
43067             Roo.get(document.head).select('link').each(function(node) { 
43068                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43069             });
43070             
43071         } else if (!this.stylesheets.length) {
43072                 // simple..
43073                 st = '<style type="text/css">' +
43074                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43075                    '</style>';
43076         } else { 
43077             st = '<style type="text/css">' +
43078                     this.stylesheets +
43079                 '</style>';
43080         }
43081         
43082         st +=  '<style type="text/css">' +
43083             'IMG { cursor: pointer } ' +
43084         '</style>';
43085
43086         var cls = 'roo-htmleditor-body';
43087         
43088         if(this.bodyCls.length){
43089             cls += ' ' + this.bodyCls;
43090         }
43091         
43092         return '<html><head>' + st  +
43093             //<style type="text/css">' +
43094             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43095             //'</style>' +
43096             ' </head><body class="' +  cls + '"></body></html>';
43097     },
43098
43099     // private
43100     onRender : function(ct, position)
43101     {
43102         var _t = this;
43103         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43104         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43105         
43106         
43107         this.el.dom.style.border = '0 none';
43108         this.el.dom.setAttribute('tabIndex', -1);
43109         this.el.addClass('x-hidden hide');
43110         
43111         
43112         
43113         if(Roo.isIE){ // fix IE 1px bogus margin
43114             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43115         }
43116        
43117         
43118         this.frameId = Roo.id();
43119         
43120          
43121         
43122         var iframe = this.owner.wrap.createChild({
43123             tag: 'iframe',
43124             cls: 'form-control', // bootstrap..
43125             id: this.frameId,
43126             name: this.frameId,
43127             frameBorder : 'no',
43128             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43129         }, this.el
43130         );
43131         
43132         
43133         this.iframe = iframe.dom;
43134
43135          this.assignDocWin();
43136         
43137         this.doc.designMode = 'on';
43138        
43139         this.doc.open();
43140         this.doc.write(this.getDocMarkup());
43141         this.doc.close();
43142
43143         
43144         var task = { // must defer to wait for browser to be ready
43145             run : function(){
43146                 //console.log("run task?" + this.doc.readyState);
43147                 this.assignDocWin();
43148                 if(this.doc.body || this.doc.readyState == 'complete'){
43149                     try {
43150                         this.doc.designMode="on";
43151                     } catch (e) {
43152                         return;
43153                     }
43154                     Roo.TaskMgr.stop(task);
43155                     this.initEditor.defer(10, this);
43156                 }
43157             },
43158             interval : 10,
43159             duration: 10000,
43160             scope: this
43161         };
43162         Roo.TaskMgr.start(task);
43163
43164     },
43165
43166     // private
43167     onResize : function(w, h)
43168     {
43169          Roo.log('resize: ' +w + ',' + h );
43170         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43171         if(!this.iframe){
43172             return;
43173         }
43174         if(typeof w == 'number'){
43175             
43176             this.iframe.style.width = w + 'px';
43177         }
43178         if(typeof h == 'number'){
43179             
43180             this.iframe.style.height = h + 'px';
43181             if(this.doc){
43182                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43183             }
43184         }
43185         
43186     },
43187
43188     /**
43189      * Toggles the editor between standard and source edit mode.
43190      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43191      */
43192     toggleSourceEdit : function(sourceEditMode){
43193         
43194         this.sourceEditMode = sourceEditMode === true;
43195         
43196         if(this.sourceEditMode){
43197  
43198             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43199             
43200         }else{
43201             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43202             //this.iframe.className = '';
43203             this.deferFocus();
43204         }
43205         //this.setSize(this.owner.wrap.getSize());
43206         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43207     },
43208
43209     
43210   
43211
43212     /**
43213      * Protected method that will not generally be called directly. If you need/want
43214      * custom HTML cleanup, this is the method you should override.
43215      * @param {String} html The HTML to be cleaned
43216      * return {String} The cleaned HTML
43217      */
43218     cleanHtml : function(html){
43219         html = String(html);
43220         if(html.length > 5){
43221             if(Roo.isSafari){ // strip safari nonsense
43222                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43223             }
43224         }
43225         if(html == '&nbsp;'){
43226             html = '';
43227         }
43228         return html;
43229     },
43230
43231     /**
43232      * HTML Editor -> Textarea
43233      * Protected method that will not generally be called directly. Syncs the contents
43234      * of the editor iframe with the textarea.
43235      */
43236     syncValue : function(){
43237         if(this.initialized){
43238             var bd = (this.doc.body || this.doc.documentElement);
43239             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43240             var html = bd.innerHTML;
43241             if(Roo.isSafari){
43242                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43243                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43244                 if(m && m[1]){
43245                     html = '<div style="'+m[0]+'">' + html + '</div>';
43246                 }
43247             }
43248             html = this.cleanHtml(html);
43249             // fix up the special chars.. normaly like back quotes in word...
43250             // however we do not want to do this with chinese..
43251             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43252                 
43253                 var cc = match.charCodeAt();
43254
43255                 // Get the character value, handling surrogate pairs
43256                 if (match.length == 2) {
43257                     // It's a surrogate pair, calculate the Unicode code point
43258                     var high = match.charCodeAt(0) - 0xD800;
43259                     var low  = match.charCodeAt(1) - 0xDC00;
43260                     cc = (high * 0x400) + low + 0x10000;
43261                 }  else if (
43262                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43263                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43264                     (cc >= 0xf900 && cc < 0xfb00 )
43265                 ) {
43266                         return match;
43267                 }  
43268          
43269                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43270                 return "&#" + cc + ";";
43271                 
43272                 
43273             });
43274             
43275             
43276              
43277             if(this.owner.fireEvent('beforesync', this, html) !== false){
43278                 this.el.dom.value = html;
43279                 this.owner.fireEvent('sync', this, html);
43280             }
43281         }
43282     },
43283
43284     /**
43285      * Protected method that will not generally be called directly. Pushes the value of the textarea
43286      * into the iframe editor.
43287      */
43288     pushValue : function(){
43289         if(this.initialized){
43290             var v = this.el.dom.value.trim();
43291             
43292 //            if(v.length < 1){
43293 //                v = '&#160;';
43294 //            }
43295             
43296             if(this.owner.fireEvent('beforepush', this, v) !== false){
43297                 var d = (this.doc.body || this.doc.documentElement);
43298                 d.innerHTML = v;
43299                 this.cleanUpPaste();
43300                 this.el.dom.value = d.innerHTML;
43301                 this.owner.fireEvent('push', this, v);
43302             }
43303         }
43304     },
43305
43306     // private
43307     deferFocus : function(){
43308         this.focus.defer(10, this);
43309     },
43310
43311     // doc'ed in Field
43312     focus : function(){
43313         if(this.win && !this.sourceEditMode){
43314             this.win.focus();
43315         }else{
43316             this.el.focus();
43317         }
43318     },
43319     
43320     assignDocWin: function()
43321     {
43322         var iframe = this.iframe;
43323         
43324          if(Roo.isIE){
43325             this.doc = iframe.contentWindow.document;
43326             this.win = iframe.contentWindow;
43327         } else {
43328 //            if (!Roo.get(this.frameId)) {
43329 //                return;
43330 //            }
43331 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43332 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43333             
43334             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43335                 return;
43336             }
43337             
43338             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43339             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43340         }
43341     },
43342     
43343     // private
43344     initEditor : function(){
43345         //console.log("INIT EDITOR");
43346         this.assignDocWin();
43347         
43348         
43349         
43350         this.doc.designMode="on";
43351         this.doc.open();
43352         this.doc.write(this.getDocMarkup());
43353         this.doc.close();
43354         
43355         var dbody = (this.doc.body || this.doc.documentElement);
43356         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43357         // this copies styles from the containing element into thsi one..
43358         // not sure why we need all of this..
43359         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43360         
43361         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43362         //ss['background-attachment'] = 'fixed'; // w3c
43363         dbody.bgProperties = 'fixed'; // ie
43364         //Roo.DomHelper.applyStyles(dbody, ss);
43365         Roo.EventManager.on(this.doc, {
43366             //'mousedown': this.onEditorEvent,
43367             'mouseup': this.onEditorEvent,
43368             'dblclick': this.onEditorEvent,
43369             'click': this.onEditorEvent,
43370             'keyup': this.onEditorEvent,
43371             buffer:100,
43372             scope: this
43373         });
43374         if(Roo.isGecko){
43375             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43376         }
43377         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43378             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43379         }
43380         this.initialized = true;
43381
43382         this.owner.fireEvent('initialize', this);
43383         this.pushValue();
43384     },
43385
43386     // private
43387     onDestroy : function(){
43388         
43389         
43390         
43391         if(this.rendered){
43392             
43393             //for (var i =0; i < this.toolbars.length;i++) {
43394             //    // fixme - ask toolbars for heights?
43395             //    this.toolbars[i].onDestroy();
43396            // }
43397             
43398             //this.wrap.dom.innerHTML = '';
43399             //this.wrap.remove();
43400         }
43401     },
43402
43403     // private
43404     onFirstFocus : function(){
43405         
43406         this.assignDocWin();
43407         
43408         
43409         this.activated = true;
43410          
43411     
43412         if(Roo.isGecko){ // prevent silly gecko errors
43413             this.win.focus();
43414             var s = this.win.getSelection();
43415             if(!s.focusNode || s.focusNode.nodeType != 3){
43416                 var r = s.getRangeAt(0);
43417                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43418                 r.collapse(true);
43419                 this.deferFocus();
43420             }
43421             try{
43422                 this.execCmd('useCSS', true);
43423                 this.execCmd('styleWithCSS', false);
43424             }catch(e){}
43425         }
43426         this.owner.fireEvent('activate', this);
43427     },
43428
43429     // private
43430     adjustFont: function(btn){
43431         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43432         //if(Roo.isSafari){ // safari
43433         //    adjust *= 2;
43434        // }
43435         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43436         if(Roo.isSafari){ // safari
43437             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43438             v =  (v < 10) ? 10 : v;
43439             v =  (v > 48) ? 48 : v;
43440             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43441             
43442         }
43443         
43444         
43445         v = Math.max(1, v+adjust);
43446         
43447         this.execCmd('FontSize', v  );
43448     },
43449
43450     onEditorEvent : function(e)
43451     {
43452         this.owner.fireEvent('editorevent', this, e);
43453       //  this.updateToolbar();
43454         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43455     },
43456
43457     insertTag : function(tg)
43458     {
43459         // could be a bit smarter... -> wrap the current selected tRoo..
43460         if (tg.toLowerCase() == 'span' || tg.toLowerCase() == 'code') {
43461             
43462             range = this.createRange(this.getSelection());
43463             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43464             wrappingNode.appendChild(range.extractContents());
43465             range.insertNode(wrappingNode);
43466
43467             return;
43468             
43469             
43470             
43471         }
43472         this.execCmd("formatblock",   tg);
43473         
43474     },
43475     
43476     insertText : function(txt)
43477     {
43478         
43479         
43480         var range = this.createRange();
43481         range.deleteContents();
43482                //alert(Sender.getAttribute('label'));
43483                
43484         range.insertNode(this.doc.createTextNode(txt));
43485     } ,
43486     
43487      
43488
43489     /**
43490      * Executes a Midas editor command on the editor document and performs necessary focus and
43491      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43492      * @param {String} cmd The Midas command
43493      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43494      */
43495     relayCmd : function(cmd, value){
43496         this.win.focus();
43497         this.execCmd(cmd, value);
43498         this.owner.fireEvent('editorevent', this);
43499         //this.updateToolbar();
43500         this.owner.deferFocus();
43501     },
43502
43503     /**
43504      * Executes a Midas editor command directly on the editor document.
43505      * For visual commands, you should use {@link #relayCmd} instead.
43506      * <b>This should only be called after the editor is initialized.</b>
43507      * @param {String} cmd The Midas command
43508      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43509      */
43510     execCmd : function(cmd, value){
43511         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43512         this.syncValue();
43513     },
43514  
43515  
43516    
43517     /**
43518      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43519      * to insert tRoo.
43520      * @param {String} text | dom node.. 
43521      */
43522     insertAtCursor : function(text)
43523     {
43524         
43525         if(!this.activated){
43526             return;
43527         }
43528         /*
43529         if(Roo.isIE){
43530             this.win.focus();
43531             var r = this.doc.selection.createRange();
43532             if(r){
43533                 r.collapse(true);
43534                 r.pasteHTML(text);
43535                 this.syncValue();
43536                 this.deferFocus();
43537             
43538             }
43539             return;
43540         }
43541         */
43542         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43543             this.win.focus();
43544             
43545             
43546             // from jquery ui (MIT licenced)
43547             var range, node;
43548             var win = this.win;
43549             
43550             if (win.getSelection && win.getSelection().getRangeAt) {
43551                 range = win.getSelection().getRangeAt(0);
43552                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43553                 range.insertNode(node);
43554             } else if (win.document.selection && win.document.selection.createRange) {
43555                 // no firefox support
43556                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43557                 win.document.selection.createRange().pasteHTML(txt);
43558             } else {
43559                 // no firefox support
43560                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43561                 this.execCmd('InsertHTML', txt);
43562             } 
43563             
43564             this.syncValue();
43565             
43566             this.deferFocus();
43567         }
43568     },
43569  // private
43570     mozKeyPress : function(e){
43571         if(e.ctrlKey){
43572             var c = e.getCharCode(), cmd;
43573           
43574             if(c > 0){
43575                 c = String.fromCharCode(c).toLowerCase();
43576                 switch(c){
43577                     case 'b':
43578                         cmd = 'bold';
43579                         break;
43580                     case 'i':
43581                         cmd = 'italic';
43582                         break;
43583                     
43584                     case 'u':
43585                         cmd = 'underline';
43586                         break;
43587                     
43588                     case 'v':
43589                         this.cleanUpPaste.defer(100, this);
43590                         return;
43591                         
43592                 }
43593                 if(cmd){
43594                     this.win.focus();
43595                     this.execCmd(cmd);
43596                     this.deferFocus();
43597                     e.preventDefault();
43598                 }
43599                 
43600             }
43601         }
43602     },
43603
43604     // private
43605     fixKeys : function(){ // load time branching for fastest keydown performance
43606         if(Roo.isIE){
43607             return function(e){
43608                 var k = e.getKey(), r;
43609                 if(k == e.TAB){
43610                     e.stopEvent();
43611                     r = this.doc.selection.createRange();
43612                     if(r){
43613                         r.collapse(true);
43614                         r.pasteHTML('&#160;&#160;&#160;&#160;');
43615                         this.deferFocus();
43616                     }
43617                     return;
43618                 }
43619                 
43620                 if(k == e.ENTER){
43621                     r = this.doc.selection.createRange();
43622                     if(r){
43623                         var target = r.parentElement();
43624                         if(!target || target.tagName.toLowerCase() != 'li'){
43625                             e.stopEvent();
43626                             r.pasteHTML('<br />');
43627                             r.collapse(false);
43628                             r.select();
43629                         }
43630                     }
43631                 }
43632                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43633                     this.cleanUpPaste.defer(100, this);
43634                     return;
43635                 }
43636                 
43637                 
43638             };
43639         }else if(Roo.isOpera){
43640             return function(e){
43641                 var k = e.getKey();
43642                 if(k == e.TAB){
43643                     e.stopEvent();
43644                     this.win.focus();
43645                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
43646                     this.deferFocus();
43647                 }
43648                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43649                     this.cleanUpPaste.defer(100, this);
43650                     return;
43651                 }
43652                 
43653             };
43654         }else if(Roo.isSafari){
43655             return function(e){
43656                 var k = e.getKey();
43657                 
43658                 if(k == e.TAB){
43659                     e.stopEvent();
43660                     this.execCmd('InsertText','\t');
43661                     this.deferFocus();
43662                     return;
43663                 }
43664                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43665                     this.cleanUpPaste.defer(100, this);
43666                     return;
43667                 }
43668                 
43669              };
43670         }
43671     }(),
43672     
43673     getAllAncestors: function()
43674     {
43675         var p = this.getSelectedNode();
43676         var a = [];
43677         if (!p) {
43678             a.push(p); // push blank onto stack..
43679             p = this.getParentElement();
43680         }
43681         
43682         
43683         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
43684             a.push(p);
43685             p = p.parentNode;
43686         }
43687         a.push(this.doc.body);
43688         return a;
43689     },
43690     lastSel : false,
43691     lastSelNode : false,
43692     
43693     
43694     getSelection : function() 
43695     {
43696         this.assignDocWin();
43697         return Roo.isIE ? this.doc.selection : this.win.getSelection();
43698     },
43699     
43700     getSelectedNode: function() 
43701     {
43702         // this may only work on Gecko!!!
43703         
43704         // should we cache this!!!!
43705         
43706         
43707         
43708          
43709         var range = this.createRange(this.getSelection()).cloneRange();
43710         
43711         if (Roo.isIE) {
43712             var parent = range.parentElement();
43713             while (true) {
43714                 var testRange = range.duplicate();
43715                 testRange.moveToElementText(parent);
43716                 if (testRange.inRange(range)) {
43717                     break;
43718                 }
43719                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
43720                     break;
43721                 }
43722                 parent = parent.parentElement;
43723             }
43724             return parent;
43725         }
43726         
43727         // is ancestor a text element.
43728         var ac =  range.commonAncestorContainer;
43729         if (ac.nodeType == 3) {
43730             ac = ac.parentNode;
43731         }
43732         
43733         var ar = ac.childNodes;
43734          
43735         var nodes = [];
43736         var other_nodes = [];
43737         var has_other_nodes = false;
43738         for (var i=0;i<ar.length;i++) {
43739             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
43740                 continue;
43741             }
43742             // fullly contained node.
43743             
43744             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
43745                 nodes.push(ar[i]);
43746                 continue;
43747             }
43748             
43749             // probably selected..
43750             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
43751                 other_nodes.push(ar[i]);
43752                 continue;
43753             }
43754             // outer..
43755             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
43756                 continue;
43757             }
43758             
43759             
43760             has_other_nodes = true;
43761         }
43762         if (!nodes.length && other_nodes.length) {
43763             nodes= other_nodes;
43764         }
43765         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
43766             return false;
43767         }
43768         
43769         return nodes[0];
43770     },
43771     createRange: function(sel)
43772     {
43773         // this has strange effects when using with 
43774         // top toolbar - not sure if it's a great idea.
43775         //this.editor.contentWindow.focus();
43776         if (typeof sel != "undefined") {
43777             try {
43778                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
43779             } catch(e) {
43780                 return this.doc.createRange();
43781             }
43782         } else {
43783             return this.doc.createRange();
43784         }
43785     },
43786     getParentElement: function()
43787     {
43788         
43789         this.assignDocWin();
43790         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
43791         
43792         var range = this.createRange(sel);
43793          
43794         try {
43795             var p = range.commonAncestorContainer;
43796             while (p.nodeType == 3) { // text node
43797                 p = p.parentNode;
43798             }
43799             return p;
43800         } catch (e) {
43801             return null;
43802         }
43803     
43804     },
43805     /***
43806      *
43807      * Range intersection.. the hard stuff...
43808      *  '-1' = before
43809      *  '0' = hits..
43810      *  '1' = after.
43811      *         [ -- selected range --- ]
43812      *   [fail]                        [fail]
43813      *
43814      *    basically..
43815      *      if end is before start or  hits it. fail.
43816      *      if start is after end or hits it fail.
43817      *
43818      *   if either hits (but other is outside. - then it's not 
43819      *   
43820      *    
43821      **/
43822     
43823     
43824     // @see http://www.thismuchiknow.co.uk/?p=64.
43825     rangeIntersectsNode : function(range, node)
43826     {
43827         var nodeRange = node.ownerDocument.createRange();
43828         try {
43829             nodeRange.selectNode(node);
43830         } catch (e) {
43831             nodeRange.selectNodeContents(node);
43832         }
43833     
43834         var rangeStartRange = range.cloneRange();
43835         rangeStartRange.collapse(true);
43836     
43837         var rangeEndRange = range.cloneRange();
43838         rangeEndRange.collapse(false);
43839     
43840         var nodeStartRange = nodeRange.cloneRange();
43841         nodeStartRange.collapse(true);
43842     
43843         var nodeEndRange = nodeRange.cloneRange();
43844         nodeEndRange.collapse(false);
43845     
43846         return rangeStartRange.compareBoundaryPoints(
43847                  Range.START_TO_START, nodeEndRange) == -1 &&
43848                rangeEndRange.compareBoundaryPoints(
43849                  Range.START_TO_START, nodeStartRange) == 1;
43850         
43851          
43852     },
43853     rangeCompareNode : function(range, node)
43854     {
43855         var nodeRange = node.ownerDocument.createRange();
43856         try {
43857             nodeRange.selectNode(node);
43858         } catch (e) {
43859             nodeRange.selectNodeContents(node);
43860         }
43861         
43862         
43863         range.collapse(true);
43864     
43865         nodeRange.collapse(true);
43866      
43867         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
43868         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
43869          
43870         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
43871         
43872         var nodeIsBefore   =  ss == 1;
43873         var nodeIsAfter    = ee == -1;
43874         
43875         if (nodeIsBefore && nodeIsAfter) {
43876             return 0; // outer
43877         }
43878         if (!nodeIsBefore && nodeIsAfter) {
43879             return 1; //right trailed.
43880         }
43881         
43882         if (nodeIsBefore && !nodeIsAfter) {
43883             return 2;  // left trailed.
43884         }
43885         // fully contined.
43886         return 3;
43887     },
43888
43889     // private? - in a new class?
43890     cleanUpPaste :  function()
43891     {
43892         // cleans up the whole document..
43893         Roo.log('cleanuppaste');
43894         
43895         this.cleanUpChildren(this.doc.body);
43896         var clean = this.cleanWordChars(this.doc.body.innerHTML);
43897         if (clean != this.doc.body.innerHTML) {
43898             this.doc.body.innerHTML = clean;
43899         }
43900         
43901     },
43902     
43903     cleanWordChars : function(input) {// change the chars to hex code
43904         var he = Roo.HtmlEditorCore;
43905         
43906         var output = input;
43907         Roo.each(he.swapCodes, function(sw) { 
43908             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
43909             
43910             output = output.replace(swapper, sw[1]);
43911         });
43912         
43913         return output;
43914     },
43915     
43916     
43917     cleanUpChildren : function (n)
43918     {
43919         if (!n.childNodes.length) {
43920             return;
43921         }
43922         for (var i = n.childNodes.length-1; i > -1 ; i--) {
43923            this.cleanUpChild(n.childNodes[i]);
43924         }
43925     },
43926     
43927     
43928         
43929     
43930     cleanUpChild : function (node)
43931     {
43932         var ed = this;
43933         //console.log(node);
43934         if (node.nodeName == "#text") {
43935             // clean up silly Windows -- stuff?
43936             return; 
43937         }
43938         if (node.nodeName == "#comment") {
43939             node.parentNode.removeChild(node);
43940             // clean up silly Windows -- stuff?
43941             return; 
43942         }
43943         var lcname = node.tagName.toLowerCase();
43944         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
43945         // whitelist of tags..
43946         
43947         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
43948             // remove node.
43949             node.parentNode.removeChild(node);
43950             return;
43951             
43952         }
43953         
43954         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
43955         
43956         // spans with no attributes - just remove them..
43957         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
43958             remove_keep_children = true;
43959         }
43960         
43961         // remove <a name=....> as rendering on yahoo mailer is borked with this.
43962         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
43963         
43964         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
43965         //    remove_keep_children = true;
43966         //}
43967         
43968         if (remove_keep_children) {
43969             this.cleanUpChildren(node);
43970             // inserts everything just before this node...
43971             while (node.childNodes.length) {
43972                 var cn = node.childNodes[0];
43973                 node.removeChild(cn);
43974                 node.parentNode.insertBefore(cn, node);
43975             }
43976             node.parentNode.removeChild(node);
43977             return;
43978         }
43979         
43980         if (!node.attributes || !node.attributes.length) {
43981             
43982           
43983             
43984             
43985             this.cleanUpChildren(node);
43986             return;
43987         }
43988         
43989         function cleanAttr(n,v)
43990         {
43991             
43992             if (v.match(/^\./) || v.match(/^\//)) {
43993                 return;
43994             }
43995             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
43996                 return;
43997             }
43998             if (v.match(/^#/)) {
43999                 return;
44000             }
44001 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44002             node.removeAttribute(n);
44003             
44004         }
44005         
44006         var cwhite = this.cwhite;
44007         var cblack = this.cblack;
44008             
44009         function cleanStyle(n,v)
44010         {
44011             if (v.match(/expression/)) { //XSS?? should we even bother..
44012                 node.removeAttribute(n);
44013                 return;
44014             }
44015             
44016             var parts = v.split(/;/);
44017             var clean = [];
44018             
44019             Roo.each(parts, function(p) {
44020                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44021                 if (!p.length) {
44022                     return true;
44023                 }
44024                 var l = p.split(':').shift().replace(/\s+/g,'');
44025                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44026                 
44027                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44028 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44029                     //node.removeAttribute(n);
44030                     return true;
44031                 }
44032                 //Roo.log()
44033                 // only allow 'c whitelisted system attributes'
44034                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44035 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44036                     //node.removeAttribute(n);
44037                     return true;
44038                 }
44039                 
44040                 
44041                  
44042                 
44043                 clean.push(p);
44044                 return true;
44045             });
44046             if (clean.length) { 
44047                 node.setAttribute(n, clean.join(';'));
44048             } else {
44049                 node.removeAttribute(n);
44050             }
44051             
44052         }
44053         
44054         
44055         for (var i = node.attributes.length-1; i > -1 ; i--) {
44056             var a = node.attributes[i];
44057             //console.log(a);
44058             
44059             if (a.name.toLowerCase().substr(0,2)=='on')  {
44060                 node.removeAttribute(a.name);
44061                 continue;
44062             }
44063             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44064                 node.removeAttribute(a.name);
44065                 continue;
44066             }
44067             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44068                 cleanAttr(a.name,a.value); // fixme..
44069                 continue;
44070             }
44071             if (a.name == 'style') {
44072                 cleanStyle(a.name,a.value);
44073                 continue;
44074             }
44075             /// clean up MS crap..
44076             // tecnically this should be a list of valid class'es..
44077             
44078             
44079             if (a.name == 'class') {
44080                 if (a.value.match(/^Mso/)) {
44081                     node.removeAttribute('class');
44082                 }
44083                 
44084                 if (a.value.match(/^body$/)) {
44085                     node.removeAttribute('class');
44086                 }
44087                 continue;
44088             }
44089             
44090             // style cleanup!?
44091             // class cleanup?
44092             
44093         }
44094         
44095         
44096         this.cleanUpChildren(node);
44097         
44098         
44099     },
44100     
44101     /**
44102      * Clean up MS wordisms...
44103      */
44104     cleanWord : function(node)
44105     {
44106         if (!node) {
44107             this.cleanWord(this.doc.body);
44108             return;
44109         }
44110         
44111         if(
44112                 node.nodeName == 'SPAN' &&
44113                 !node.hasAttributes() &&
44114                 node.childNodes.length == 1 &&
44115                 node.firstChild.nodeName == "#text"  
44116         ) {
44117             var textNode = node.firstChild;
44118             node.removeChild(textNode);
44119             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44120                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44121             }
44122             node.parentNode.insertBefore(textNode, node);
44123             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44124                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44125             }
44126             node.parentNode.removeChild(node);
44127         }
44128         
44129         if (node.nodeName == "#text") {
44130             // clean up silly Windows -- stuff?
44131             return; 
44132         }
44133         if (node.nodeName == "#comment") {
44134             node.parentNode.removeChild(node);
44135             // clean up silly Windows -- stuff?
44136             return; 
44137         }
44138         
44139         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44140             node.parentNode.removeChild(node);
44141             return;
44142         }
44143         Roo.log(node.tagName);
44144         // remove - but keep children..
44145         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44146             Roo.log('-- removed');
44147             while (node.childNodes.length) {
44148                 var cn = node.childNodes[0];
44149                 node.removeChild(cn);
44150                 node.parentNode.insertBefore(cn, node);
44151             }
44152             node.parentNode.removeChild(node);
44153             /// no need to iterate chidlren = it's got none..
44154             //this.iterateChildren(node, this.cleanWord);
44155             return;
44156         }
44157         // clean styles
44158         if (node.className.length) {
44159             
44160             var cn = node.className.split(/\W+/);
44161             var cna = [];
44162             Roo.each(cn, function(cls) {
44163                 if (cls.match(/Mso[a-zA-Z]+/)) {
44164                     return;
44165                 }
44166                 cna.push(cls);
44167             });
44168             node.className = cna.length ? cna.join(' ') : '';
44169             if (!cna.length) {
44170                 node.removeAttribute("class");
44171             }
44172         }
44173         
44174         if (node.hasAttribute("lang")) {
44175             node.removeAttribute("lang");
44176         }
44177         
44178         if (node.hasAttribute("style")) {
44179             
44180             var styles = node.getAttribute("style").split(";");
44181             var nstyle = [];
44182             Roo.each(styles, function(s) {
44183                 if (!s.match(/:/)) {
44184                     return;
44185                 }
44186                 var kv = s.split(":");
44187                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44188                     return;
44189                 }
44190                 // what ever is left... we allow.
44191                 nstyle.push(s);
44192             });
44193             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44194             if (!nstyle.length) {
44195                 node.removeAttribute('style');
44196             }
44197         }
44198         this.iterateChildren(node, this.cleanWord);
44199         
44200         
44201         
44202     },
44203     /**
44204      * iterateChildren of a Node, calling fn each time, using this as the scole..
44205      * @param {DomNode} node node to iterate children of.
44206      * @param {Function} fn method of this class to call on each item.
44207      */
44208     iterateChildren : function(node, fn)
44209     {
44210         if (!node.childNodes.length) {
44211                 return;
44212         }
44213         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44214            fn.call(this, node.childNodes[i])
44215         }
44216     },
44217     
44218     
44219     /**
44220      * cleanTableWidths.
44221      *
44222      * Quite often pasting from word etc.. results in tables with column and widths.
44223      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44224      *
44225      */
44226     cleanTableWidths : function(node)
44227     {
44228          
44229          
44230         if (!node) {
44231             this.cleanTableWidths(this.doc.body);
44232             return;
44233         }
44234         
44235         // ignore list...
44236         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44237             return; 
44238         }
44239         Roo.log(node.tagName);
44240         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44241             this.iterateChildren(node, this.cleanTableWidths);
44242             return;
44243         }
44244         if (node.hasAttribute('width')) {
44245             node.removeAttribute('width');
44246         }
44247         
44248          
44249         if (node.hasAttribute("style")) {
44250             // pretty basic...
44251             
44252             var styles = node.getAttribute("style").split(";");
44253             var nstyle = [];
44254             Roo.each(styles, function(s) {
44255                 if (!s.match(/:/)) {
44256                     return;
44257                 }
44258                 var kv = s.split(":");
44259                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44260                     return;
44261                 }
44262                 // what ever is left... we allow.
44263                 nstyle.push(s);
44264             });
44265             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44266             if (!nstyle.length) {
44267                 node.removeAttribute('style');
44268             }
44269         }
44270         
44271         this.iterateChildren(node, this.cleanTableWidths);
44272         
44273         
44274     },
44275     
44276     
44277     
44278     
44279     domToHTML : function(currentElement, depth, nopadtext) {
44280         
44281         depth = depth || 0;
44282         nopadtext = nopadtext || false;
44283     
44284         if (!currentElement) {
44285             return this.domToHTML(this.doc.body);
44286         }
44287         
44288         //Roo.log(currentElement);
44289         var j;
44290         var allText = false;
44291         var nodeName = currentElement.nodeName;
44292         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44293         
44294         if  (nodeName == '#text') {
44295             
44296             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44297         }
44298         
44299         
44300         var ret = '';
44301         if (nodeName != 'BODY') {
44302              
44303             var i = 0;
44304             // Prints the node tagName, such as <A>, <IMG>, etc
44305             if (tagName) {
44306                 var attr = [];
44307                 for(i = 0; i < currentElement.attributes.length;i++) {
44308                     // quoting?
44309                     var aname = currentElement.attributes.item(i).name;
44310                     if (!currentElement.attributes.item(i).value.length) {
44311                         continue;
44312                     }
44313                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44314                 }
44315                 
44316                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44317             } 
44318             else {
44319                 
44320                 // eack
44321             }
44322         } else {
44323             tagName = false;
44324         }
44325         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44326             return ret;
44327         }
44328         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44329             nopadtext = true;
44330         }
44331         
44332         
44333         // Traverse the tree
44334         i = 0;
44335         var currentElementChild = currentElement.childNodes.item(i);
44336         var allText = true;
44337         var innerHTML  = '';
44338         lastnode = '';
44339         while (currentElementChild) {
44340             // Formatting code (indent the tree so it looks nice on the screen)
44341             var nopad = nopadtext;
44342             if (lastnode == 'SPAN') {
44343                 nopad  = true;
44344             }
44345             // text
44346             if  (currentElementChild.nodeName == '#text') {
44347                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44348                 toadd = nopadtext ? toadd : toadd.trim();
44349                 if (!nopad && toadd.length > 80) {
44350                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44351                 }
44352                 innerHTML  += toadd;
44353                 
44354                 i++;
44355                 currentElementChild = currentElement.childNodes.item(i);
44356                 lastNode = '';
44357                 continue;
44358             }
44359             allText = false;
44360             
44361             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44362                 
44363             // Recursively traverse the tree structure of the child node
44364             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44365             lastnode = currentElementChild.nodeName;
44366             i++;
44367             currentElementChild=currentElement.childNodes.item(i);
44368         }
44369         
44370         ret += innerHTML;
44371         
44372         if (!allText) {
44373                 // The remaining code is mostly for formatting the tree
44374             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44375         }
44376         
44377         
44378         if (tagName) {
44379             ret+= "</"+tagName+">";
44380         }
44381         return ret;
44382         
44383     },
44384         
44385     applyBlacklists : function()
44386     {
44387         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44388         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44389         
44390         this.white = [];
44391         this.black = [];
44392         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44393             if (b.indexOf(tag) > -1) {
44394                 return;
44395             }
44396             this.white.push(tag);
44397             
44398         }, this);
44399         
44400         Roo.each(w, function(tag) {
44401             if (b.indexOf(tag) > -1) {
44402                 return;
44403             }
44404             if (this.white.indexOf(tag) > -1) {
44405                 return;
44406             }
44407             this.white.push(tag);
44408             
44409         }, this);
44410         
44411         
44412         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44413             if (w.indexOf(tag) > -1) {
44414                 return;
44415             }
44416             this.black.push(tag);
44417             
44418         }, this);
44419         
44420         Roo.each(b, function(tag) {
44421             if (w.indexOf(tag) > -1) {
44422                 return;
44423             }
44424             if (this.black.indexOf(tag) > -1) {
44425                 return;
44426             }
44427             this.black.push(tag);
44428             
44429         }, this);
44430         
44431         
44432         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44433         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44434         
44435         this.cwhite = [];
44436         this.cblack = [];
44437         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44438             if (b.indexOf(tag) > -1) {
44439                 return;
44440             }
44441             this.cwhite.push(tag);
44442             
44443         }, this);
44444         
44445         Roo.each(w, function(tag) {
44446             if (b.indexOf(tag) > -1) {
44447                 return;
44448             }
44449             if (this.cwhite.indexOf(tag) > -1) {
44450                 return;
44451             }
44452             this.cwhite.push(tag);
44453             
44454         }, this);
44455         
44456         
44457         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44458             if (w.indexOf(tag) > -1) {
44459                 return;
44460             }
44461             this.cblack.push(tag);
44462             
44463         }, this);
44464         
44465         Roo.each(b, function(tag) {
44466             if (w.indexOf(tag) > -1) {
44467                 return;
44468             }
44469             if (this.cblack.indexOf(tag) > -1) {
44470                 return;
44471             }
44472             this.cblack.push(tag);
44473             
44474         }, this);
44475     },
44476     
44477     setStylesheets : function(stylesheets)
44478     {
44479         if(typeof(stylesheets) == 'string'){
44480             Roo.get(this.iframe.contentDocument.head).createChild({
44481                 tag : 'link',
44482                 rel : 'stylesheet',
44483                 type : 'text/css',
44484                 href : stylesheets
44485             });
44486             
44487             return;
44488         }
44489         var _this = this;
44490      
44491         Roo.each(stylesheets, function(s) {
44492             if(!s.length){
44493                 return;
44494             }
44495             
44496             Roo.get(_this.iframe.contentDocument.head).createChild({
44497                 tag : 'link',
44498                 rel : 'stylesheet',
44499                 type : 'text/css',
44500                 href : s
44501             });
44502         });
44503
44504         
44505     },
44506     
44507     removeStylesheets : function()
44508     {
44509         var _this = this;
44510         
44511         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44512             s.remove();
44513         });
44514     },
44515     
44516     setStyle : function(style)
44517     {
44518         Roo.get(this.iframe.contentDocument.head).createChild({
44519             tag : 'style',
44520             type : 'text/css',
44521             html : style
44522         });
44523
44524         return;
44525     }
44526     
44527     // hide stuff that is not compatible
44528     /**
44529      * @event blur
44530      * @hide
44531      */
44532     /**
44533      * @event change
44534      * @hide
44535      */
44536     /**
44537      * @event focus
44538      * @hide
44539      */
44540     /**
44541      * @event specialkey
44542      * @hide
44543      */
44544     /**
44545      * @cfg {String} fieldClass @hide
44546      */
44547     /**
44548      * @cfg {String} focusClass @hide
44549      */
44550     /**
44551      * @cfg {String} autoCreate @hide
44552      */
44553     /**
44554      * @cfg {String} inputType @hide
44555      */
44556     /**
44557      * @cfg {String} invalidClass @hide
44558      */
44559     /**
44560      * @cfg {String} invalidText @hide
44561      */
44562     /**
44563      * @cfg {String} msgFx @hide
44564      */
44565     /**
44566      * @cfg {String} validateOnBlur @hide
44567      */
44568 });
44569
44570 Roo.HtmlEditorCore.white = [
44571         'area', 'br', 'img', 'input', 'hr', 'wbr',
44572         
44573        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44574        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44575        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44576        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44577        'table',   'ul',         'xmp', 
44578        
44579        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44580       'thead',   'tr', 
44581      
44582       'dir', 'menu', 'ol', 'ul', 'dl',
44583        
44584       'embed',  'object'
44585 ];
44586
44587
44588 Roo.HtmlEditorCore.black = [
44589     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44590         'applet', // 
44591         'base',   'basefont', 'bgsound', 'blink',  'body', 
44592         'frame',  'frameset', 'head',    'html',   'ilayer', 
44593         'iframe', 'layer',  'link',     'meta',    'object',   
44594         'script', 'style' ,'title',  'xml' // clean later..
44595 ];
44596 Roo.HtmlEditorCore.clean = [
44597     'script', 'style', 'title', 'xml'
44598 ];
44599 Roo.HtmlEditorCore.remove = [
44600     'font'
44601 ];
44602 // attributes..
44603
44604 Roo.HtmlEditorCore.ablack = [
44605     'on'
44606 ];
44607     
44608 Roo.HtmlEditorCore.aclean = [ 
44609     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
44610 ];
44611
44612 // protocols..
44613 Roo.HtmlEditorCore.pwhite= [
44614         'http',  'https',  'mailto'
44615 ];
44616
44617 // white listed style attributes.
44618 Roo.HtmlEditorCore.cwhite= [
44619       //  'text-align', /// default is to allow most things..
44620       
44621          
44622 //        'font-size'//??
44623 ];
44624
44625 // black listed style attributes.
44626 Roo.HtmlEditorCore.cblack= [
44627       //  'font-size' -- this can be set by the project 
44628 ];
44629
44630
44631 Roo.HtmlEditorCore.swapCodes   =[ 
44632     [    8211, "--" ], 
44633     [    8212, "--" ], 
44634     [    8216,  "'" ],  
44635     [    8217, "'" ],  
44636     [    8220, '"' ],  
44637     [    8221, '"' ],  
44638     [    8226, "*" ],  
44639     [    8230, "..." ]
44640 ]; 
44641
44642     //<script type="text/javascript">
44643
44644 /*
44645  * Ext JS Library 1.1.1
44646  * Copyright(c) 2006-2007, Ext JS, LLC.
44647  * Licence LGPL
44648  * 
44649  */
44650  
44651  
44652 Roo.form.HtmlEditor = function(config){
44653     
44654     
44655     
44656     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
44657     
44658     if (!this.toolbars) {
44659         this.toolbars = [];
44660     }
44661     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
44662     
44663     
44664 };
44665
44666 /**
44667  * @class Roo.form.HtmlEditor
44668  * @extends Roo.form.Field
44669  * Provides a lightweight HTML Editor component.
44670  *
44671  * This has been tested on Fireforx / Chrome.. IE may not be so great..
44672  * 
44673  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
44674  * supported by this editor.</b><br/><br/>
44675  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
44676  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
44677  */
44678 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
44679     /**
44680      * @cfg {Boolean} clearUp
44681      */
44682     clearUp : true,
44683       /**
44684      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
44685      */
44686     toolbars : false,
44687    
44688      /**
44689      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
44690      *                        Roo.resizable.
44691      */
44692     resizable : false,
44693      /**
44694      * @cfg {Number} height (in pixels)
44695      */   
44696     height: 300,
44697    /**
44698      * @cfg {Number} width (in pixels)
44699      */   
44700     width: 500,
44701     
44702     /**
44703      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
44704      * 
44705      */
44706     stylesheets: false,
44707     
44708     
44709      /**
44710      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
44711      * 
44712      */
44713     cblack: false,
44714     /**
44715      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
44716      * 
44717      */
44718     cwhite: false,
44719     
44720      /**
44721      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
44722      * 
44723      */
44724     black: false,
44725     /**
44726      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
44727      * 
44728      */
44729     white: false,
44730     
44731     // id of frame..
44732     frameId: false,
44733     
44734     // private properties
44735     validationEvent : false,
44736     deferHeight: true,
44737     initialized : false,
44738     activated : false,
44739     
44740     onFocus : Roo.emptyFn,
44741     iframePad:3,
44742     hideMode:'offsets',
44743     
44744     actionMode : 'container', // defaults to hiding it...
44745     
44746     defaultAutoCreate : { // modified by initCompnoent..
44747         tag: "textarea",
44748         style:"width:500px;height:300px;",
44749         autocomplete: "new-password"
44750     },
44751
44752     // private
44753     initComponent : function(){
44754         this.addEvents({
44755             /**
44756              * @event initialize
44757              * Fires when the editor is fully initialized (including the iframe)
44758              * @param {HtmlEditor} this
44759              */
44760             initialize: true,
44761             /**
44762              * @event activate
44763              * Fires when the editor is first receives the focus. Any insertion must wait
44764              * until after this event.
44765              * @param {HtmlEditor} this
44766              */
44767             activate: true,
44768              /**
44769              * @event beforesync
44770              * Fires before the textarea is updated with content from the editor iframe. Return false
44771              * to cancel the sync.
44772              * @param {HtmlEditor} this
44773              * @param {String} html
44774              */
44775             beforesync: true,
44776              /**
44777              * @event beforepush
44778              * Fires before the iframe editor is updated with content from the textarea. Return false
44779              * to cancel the push.
44780              * @param {HtmlEditor} this
44781              * @param {String} html
44782              */
44783             beforepush: true,
44784              /**
44785              * @event sync
44786              * Fires when the textarea is updated with content from the editor iframe.
44787              * @param {HtmlEditor} this
44788              * @param {String} html
44789              */
44790             sync: true,
44791              /**
44792              * @event push
44793              * Fires when the iframe editor is updated with content from the textarea.
44794              * @param {HtmlEditor} this
44795              * @param {String} html
44796              */
44797             push: true,
44798              /**
44799              * @event editmodechange
44800              * Fires when the editor switches edit modes
44801              * @param {HtmlEditor} this
44802              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
44803              */
44804             editmodechange: true,
44805             /**
44806              * @event editorevent
44807              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
44808              * @param {HtmlEditor} this
44809              */
44810             editorevent: true,
44811             /**
44812              * @event firstfocus
44813              * Fires when on first focus - needed by toolbars..
44814              * @param {HtmlEditor} this
44815              */
44816             firstfocus: true,
44817             /**
44818              * @event autosave
44819              * Auto save the htmlEditor value as a file into Events
44820              * @param {HtmlEditor} this
44821              */
44822             autosave: true,
44823             /**
44824              * @event savedpreview
44825              * preview the saved version of htmlEditor
44826              * @param {HtmlEditor} this
44827              */
44828             savedpreview: true,
44829             
44830             /**
44831             * @event stylesheetsclick
44832             * Fires when press the Sytlesheets button
44833             * @param {Roo.HtmlEditorCore} this
44834             */
44835             stylesheetsclick: true
44836         });
44837         this.defaultAutoCreate =  {
44838             tag: "textarea",
44839             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
44840             autocomplete: "new-password"
44841         };
44842     },
44843
44844     /**
44845      * Protected method that will not generally be called directly. It
44846      * is called when the editor creates its toolbar. Override this method if you need to
44847      * add custom toolbar buttons.
44848      * @param {HtmlEditor} editor
44849      */
44850     createToolbar : function(editor){
44851         Roo.log("create toolbars");
44852         if (!editor.toolbars || !editor.toolbars.length) {
44853             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
44854         }
44855         
44856         for (var i =0 ; i < editor.toolbars.length;i++) {
44857             editor.toolbars[i] = Roo.factory(
44858                     typeof(editor.toolbars[i]) == 'string' ?
44859                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
44860                 Roo.form.HtmlEditor);
44861             editor.toolbars[i].init(editor);
44862         }
44863          
44864         
44865     },
44866
44867      
44868     // private
44869     onRender : function(ct, position)
44870     {
44871         var _t = this;
44872         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
44873         
44874         this.wrap = this.el.wrap({
44875             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
44876         });
44877         
44878         this.editorcore.onRender(ct, position);
44879          
44880         if (this.resizable) {
44881             this.resizeEl = new Roo.Resizable(this.wrap, {
44882                 pinned : true,
44883                 wrap: true,
44884                 dynamic : true,
44885                 minHeight : this.height,
44886                 height: this.height,
44887                 handles : this.resizable,
44888                 width: this.width,
44889                 listeners : {
44890                     resize : function(r, w, h) {
44891                         _t.onResize(w,h); // -something
44892                     }
44893                 }
44894             });
44895             
44896         }
44897         this.createToolbar(this);
44898        
44899         
44900         if(!this.width){
44901             this.setSize(this.wrap.getSize());
44902         }
44903         if (this.resizeEl) {
44904             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
44905             // should trigger onReize..
44906         }
44907         
44908         this.keyNav = new Roo.KeyNav(this.el, {
44909             
44910             "tab" : function(e){
44911                 e.preventDefault();
44912                 
44913                 var value = this.getValue();
44914                 
44915                 var start = this.el.dom.selectionStart;
44916                 var end = this.el.dom.selectionEnd;
44917                 
44918                 if(!e.shiftKey){
44919                     
44920                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
44921                     this.el.dom.setSelectionRange(end + 1, end + 1);
44922                     return;
44923                 }
44924                 
44925                 var f = value.substring(0, start).split("\t");
44926                 
44927                 if(f.pop().length != 0){
44928                     return;
44929                 }
44930                 
44931                 this.setValue(f.join("\t") + value.substring(end));
44932                 this.el.dom.setSelectionRange(start - 1, start - 1);
44933                 
44934             },
44935             
44936             "home" : function(e){
44937                 e.preventDefault();
44938                 
44939                 var curr = this.el.dom.selectionStart;
44940                 var lines = this.getValue().split("\n");
44941                 
44942                 if(!lines.length){
44943                     return;
44944                 }
44945                 
44946                 if(e.ctrlKey){
44947                     this.el.dom.setSelectionRange(0, 0);
44948                     return;
44949                 }
44950                 
44951                 var pos = 0;
44952                 
44953                 for (var i = 0; i < lines.length;i++) {
44954                     pos += lines[i].length;
44955                     
44956                     if(i != 0){
44957                         pos += 1;
44958                     }
44959                     
44960                     if(pos < curr){
44961                         continue;
44962                     }
44963                     
44964                     pos -= lines[i].length;
44965                     
44966                     break;
44967                 }
44968                 
44969                 if(!e.shiftKey){
44970                     this.el.dom.setSelectionRange(pos, pos);
44971                     return;
44972                 }
44973                 
44974                 this.el.dom.selectionStart = pos;
44975                 this.el.dom.selectionEnd = curr;
44976             },
44977             
44978             "end" : function(e){
44979                 e.preventDefault();
44980                 
44981                 var curr = this.el.dom.selectionStart;
44982                 var lines = this.getValue().split("\n");
44983                 
44984                 if(!lines.length){
44985                     return;
44986                 }
44987                 
44988                 if(e.ctrlKey){
44989                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
44990                     return;
44991                 }
44992                 
44993                 var pos = 0;
44994                 
44995                 for (var i = 0; i < lines.length;i++) {
44996                     
44997                     pos += lines[i].length;
44998                     
44999                     if(i != 0){
45000                         pos += 1;
45001                     }
45002                     
45003                     if(pos < curr){
45004                         continue;
45005                     }
45006                     
45007                     break;
45008                 }
45009                 
45010                 if(!e.shiftKey){
45011                     this.el.dom.setSelectionRange(pos, pos);
45012                     return;
45013                 }
45014                 
45015                 this.el.dom.selectionStart = curr;
45016                 this.el.dom.selectionEnd = pos;
45017             },
45018
45019             scope : this,
45020
45021             doRelay : function(foo, bar, hname){
45022                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45023             },
45024
45025             forceKeyDown: true
45026         });
45027         
45028 //        if(this.autosave && this.w){
45029 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45030 //        }
45031     },
45032
45033     // private
45034     onResize : function(w, h)
45035     {
45036         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45037         var ew = false;
45038         var eh = false;
45039         
45040         if(this.el ){
45041             if(typeof w == 'number'){
45042                 var aw = w - this.wrap.getFrameWidth('lr');
45043                 this.el.setWidth(this.adjustWidth('textarea', aw));
45044                 ew = aw;
45045             }
45046             if(typeof h == 'number'){
45047                 var tbh = 0;
45048                 for (var i =0; i < this.toolbars.length;i++) {
45049                     // fixme - ask toolbars for heights?
45050                     tbh += this.toolbars[i].tb.el.getHeight();
45051                     if (this.toolbars[i].footer) {
45052                         tbh += this.toolbars[i].footer.el.getHeight();
45053                     }
45054                 }
45055                 
45056                 
45057                 
45058                 
45059                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45060                 ah -= 5; // knock a few pixes off for look..
45061 //                Roo.log(ah);
45062                 this.el.setHeight(this.adjustWidth('textarea', ah));
45063                 var eh = ah;
45064             }
45065         }
45066         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45067         this.editorcore.onResize(ew,eh);
45068         
45069     },
45070
45071     /**
45072      * Toggles the editor between standard and source edit mode.
45073      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45074      */
45075     toggleSourceEdit : function(sourceEditMode)
45076     {
45077         this.editorcore.toggleSourceEdit(sourceEditMode);
45078         
45079         if(this.editorcore.sourceEditMode){
45080             Roo.log('editor - showing textarea');
45081             
45082 //            Roo.log('in');
45083 //            Roo.log(this.syncValue());
45084             this.editorcore.syncValue();
45085             this.el.removeClass('x-hidden');
45086             this.el.dom.removeAttribute('tabIndex');
45087             this.el.focus();
45088             
45089             for (var i = 0; i < this.toolbars.length; i++) {
45090                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45091                     this.toolbars[i].tb.hide();
45092                     this.toolbars[i].footer.hide();
45093                 }
45094             }
45095             
45096         }else{
45097             Roo.log('editor - hiding textarea');
45098 //            Roo.log('out')
45099 //            Roo.log(this.pushValue()); 
45100             this.editorcore.pushValue();
45101             
45102             this.el.addClass('x-hidden');
45103             this.el.dom.setAttribute('tabIndex', -1);
45104             
45105             for (var i = 0; i < this.toolbars.length; i++) {
45106                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45107                     this.toolbars[i].tb.show();
45108                     this.toolbars[i].footer.show();
45109                 }
45110             }
45111             
45112             //this.deferFocus();
45113         }
45114         
45115         this.setSize(this.wrap.getSize());
45116         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45117         
45118         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45119     },
45120  
45121     // private (for BoxComponent)
45122     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45123
45124     // private (for BoxComponent)
45125     getResizeEl : function(){
45126         return this.wrap;
45127     },
45128
45129     // private (for BoxComponent)
45130     getPositionEl : function(){
45131         return this.wrap;
45132     },
45133
45134     // private
45135     initEvents : function(){
45136         this.originalValue = this.getValue();
45137     },
45138
45139     /**
45140      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45141      * @method
45142      */
45143     markInvalid : Roo.emptyFn,
45144     /**
45145      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45146      * @method
45147      */
45148     clearInvalid : Roo.emptyFn,
45149
45150     setValue : function(v){
45151         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45152         this.editorcore.pushValue();
45153     },
45154
45155      
45156     // private
45157     deferFocus : function(){
45158         this.focus.defer(10, this);
45159     },
45160
45161     // doc'ed in Field
45162     focus : function(){
45163         this.editorcore.focus();
45164         
45165     },
45166       
45167
45168     // private
45169     onDestroy : function(){
45170         
45171         
45172         
45173         if(this.rendered){
45174             
45175             for (var i =0; i < this.toolbars.length;i++) {
45176                 // fixme - ask toolbars for heights?
45177                 this.toolbars[i].onDestroy();
45178             }
45179             
45180             this.wrap.dom.innerHTML = '';
45181             this.wrap.remove();
45182         }
45183     },
45184
45185     // private
45186     onFirstFocus : function(){
45187         //Roo.log("onFirstFocus");
45188         this.editorcore.onFirstFocus();
45189          for (var i =0; i < this.toolbars.length;i++) {
45190             this.toolbars[i].onFirstFocus();
45191         }
45192         
45193     },
45194     
45195     // private
45196     syncValue : function()
45197     {
45198         this.editorcore.syncValue();
45199     },
45200     
45201     pushValue : function()
45202     {
45203         this.editorcore.pushValue();
45204     },
45205     
45206     setStylesheets : function(stylesheets)
45207     {
45208         this.editorcore.setStylesheets(stylesheets);
45209     },
45210     
45211     removeStylesheets : function()
45212     {
45213         this.editorcore.removeStylesheets();
45214     }
45215      
45216     
45217     // hide stuff that is not compatible
45218     /**
45219      * @event blur
45220      * @hide
45221      */
45222     /**
45223      * @event change
45224      * @hide
45225      */
45226     /**
45227      * @event focus
45228      * @hide
45229      */
45230     /**
45231      * @event specialkey
45232      * @hide
45233      */
45234     /**
45235      * @cfg {String} fieldClass @hide
45236      */
45237     /**
45238      * @cfg {String} focusClass @hide
45239      */
45240     /**
45241      * @cfg {String} autoCreate @hide
45242      */
45243     /**
45244      * @cfg {String} inputType @hide
45245      */
45246     /**
45247      * @cfg {String} invalidClass @hide
45248      */
45249     /**
45250      * @cfg {String} invalidText @hide
45251      */
45252     /**
45253      * @cfg {String} msgFx @hide
45254      */
45255     /**
45256      * @cfg {String} validateOnBlur @hide
45257      */
45258 });
45259  
45260     // <script type="text/javascript">
45261 /*
45262  * Based on
45263  * Ext JS Library 1.1.1
45264  * Copyright(c) 2006-2007, Ext JS, LLC.
45265  *  
45266  
45267  */
45268
45269 /**
45270  * @class Roo.form.HtmlEditorToolbar1
45271  * Basic Toolbar
45272  * 
45273  * Usage:
45274  *
45275  new Roo.form.HtmlEditor({
45276     ....
45277     toolbars : [
45278         new Roo.form.HtmlEditorToolbar1({
45279             disable : { fonts: 1 , format: 1, ..., ... , ...],
45280             btns : [ .... ]
45281         })
45282     }
45283      
45284  * 
45285  * @cfg {Object} disable List of elements to disable..
45286  * @cfg {Array} btns List of additional buttons.
45287  * 
45288  * 
45289  * NEEDS Extra CSS? 
45290  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45291  */
45292  
45293 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45294 {
45295     
45296     Roo.apply(this, config);
45297     
45298     // default disabled, based on 'good practice'..
45299     this.disable = this.disable || {};
45300     Roo.applyIf(this.disable, {
45301         fontSize : true,
45302         colors : true,
45303         specialElements : true
45304     });
45305     
45306     
45307     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45308     // dont call parent... till later.
45309 }
45310
45311 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45312     
45313     tb: false,
45314     
45315     rendered: false,
45316     
45317     editor : false,
45318     editorcore : false,
45319     /**
45320      * @cfg {Object} disable  List of toolbar elements to disable
45321          
45322      */
45323     disable : false,
45324     
45325     
45326      /**
45327      * @cfg {String} createLinkText The default text for the create link prompt
45328      */
45329     createLinkText : 'Please enter the URL for the link:',
45330     /**
45331      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45332      */
45333     defaultLinkValue : 'http:/'+'/',
45334    
45335     
45336       /**
45337      * @cfg {Array} fontFamilies An array of available font families
45338      */
45339     fontFamilies : [
45340         'Arial',
45341         'Courier New',
45342         'Tahoma',
45343         'Times New Roman',
45344         'Verdana'
45345     ],
45346     
45347     specialChars : [
45348            "&#169;",
45349           "&#174;",     
45350           "&#8482;",    
45351           "&#163;" ,    
45352          // "&#8212;",    
45353           "&#8230;",    
45354           "&#247;" ,    
45355         //  "&#225;" ,     ?? a acute?
45356            "&#8364;"    , //Euro
45357        //   "&#8220;"    ,
45358         //  "&#8221;"    ,
45359         //  "&#8226;"    ,
45360           "&#176;"  //   , // degrees
45361
45362          // "&#233;"     , // e ecute
45363          // "&#250;"     , // u ecute?
45364     ],
45365     
45366     specialElements : [
45367         {
45368             text: "Insert Table",
45369             xtype: 'MenuItem',
45370             xns : Roo.Menu,
45371             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45372                 
45373         },
45374         {    
45375             text: "Insert Image",
45376             xtype: 'MenuItem',
45377             xns : Roo.Menu,
45378             ihtml : '<img src="about:blank"/>'
45379             
45380         }
45381         
45382          
45383     ],
45384     
45385     
45386     inputElements : [ 
45387             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45388             "input:submit", "input:button", "select", "textarea", "label" ],
45389     formats : [
45390         ["p"] ,  
45391         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45392         ["pre"],[ "code"], 
45393         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45394         ['div'],['span']
45395     ],
45396     
45397     cleanStyles : [
45398         "font-size"
45399     ],
45400      /**
45401      * @cfg {String} defaultFont default font to use.
45402      */
45403     defaultFont: 'tahoma',
45404    
45405     fontSelect : false,
45406     
45407     
45408     formatCombo : false,
45409     
45410     init : function(editor)
45411     {
45412         this.editor = editor;
45413         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45414         var editorcore = this.editorcore;
45415         
45416         var _t = this;
45417         
45418         var fid = editorcore.frameId;
45419         var etb = this;
45420         function btn(id, toggle, handler){
45421             var xid = fid + '-'+ id ;
45422             return {
45423                 id : xid,
45424                 cmd : id,
45425                 cls : 'x-btn-icon x-edit-'+id,
45426                 enableToggle:toggle !== false,
45427                 scope: _t, // was editor...
45428                 handler:handler||_t.relayBtnCmd,
45429                 clickEvent:'mousedown',
45430                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45431                 tabIndex:-1
45432             };
45433         }
45434         
45435         
45436         
45437         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45438         this.tb = tb;
45439          // stop form submits
45440         tb.el.on('click', function(e){
45441             e.preventDefault(); // what does this do?
45442         });
45443
45444         if(!this.disable.font) { // && !Roo.isSafari){
45445             /* why no safari for fonts 
45446             editor.fontSelect = tb.el.createChild({
45447                 tag:'select',
45448                 tabIndex: -1,
45449                 cls:'x-font-select',
45450                 html: this.createFontOptions()
45451             });
45452             
45453             editor.fontSelect.on('change', function(){
45454                 var font = editor.fontSelect.dom.value;
45455                 editor.relayCmd('fontname', font);
45456                 editor.deferFocus();
45457             }, editor);
45458             
45459             tb.add(
45460                 editor.fontSelect.dom,
45461                 '-'
45462             );
45463             */
45464             
45465         };
45466         if(!this.disable.formats){
45467             this.formatCombo = new Roo.form.ComboBox({
45468                 store: new Roo.data.SimpleStore({
45469                     id : 'tag',
45470                     fields: ['tag'],
45471                     data : this.formats // from states.js
45472                 }),
45473                 blockFocus : true,
45474                 name : '',
45475                 //autoCreate : {tag: "div",  size: "20"},
45476                 displayField:'tag',
45477                 typeAhead: false,
45478                 mode: 'local',
45479                 editable : false,
45480                 triggerAction: 'all',
45481                 emptyText:'Add tag',
45482                 selectOnFocus:true,
45483                 width:135,
45484                 listeners : {
45485                     'select': function(c, r, i) {
45486                         editorcore.insertTag(r.get('tag'));
45487                         editor.focus();
45488                     }
45489                 }
45490
45491             });
45492             tb.addField(this.formatCombo);
45493             
45494         }
45495         
45496         if(!this.disable.format){
45497             tb.add(
45498                 btn('bold'),
45499                 btn('italic'),
45500                 btn('underline'),
45501                 btn('strikethrough')
45502             );
45503         };
45504         if(!this.disable.fontSize){
45505             tb.add(
45506                 '-',
45507                 
45508                 
45509                 btn('increasefontsize', false, editorcore.adjustFont),
45510                 btn('decreasefontsize', false, editorcore.adjustFont)
45511             );
45512         };
45513         
45514         
45515         if(!this.disable.colors){
45516             tb.add(
45517                 '-', {
45518                     id:editorcore.frameId +'-forecolor',
45519                     cls:'x-btn-icon x-edit-forecolor',
45520                     clickEvent:'mousedown',
45521                     tooltip: this.buttonTips['forecolor'] || undefined,
45522                     tabIndex:-1,
45523                     menu : new Roo.menu.ColorMenu({
45524                         allowReselect: true,
45525                         focus: Roo.emptyFn,
45526                         value:'000000',
45527                         plain:true,
45528                         selectHandler: function(cp, color){
45529                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45530                             editor.deferFocus();
45531                         },
45532                         scope: editorcore,
45533                         clickEvent:'mousedown'
45534                     })
45535                 }, {
45536                     id:editorcore.frameId +'backcolor',
45537                     cls:'x-btn-icon x-edit-backcolor',
45538                     clickEvent:'mousedown',
45539                     tooltip: this.buttonTips['backcolor'] || undefined,
45540                     tabIndex:-1,
45541                     menu : new Roo.menu.ColorMenu({
45542                         focus: Roo.emptyFn,
45543                         value:'FFFFFF',
45544                         plain:true,
45545                         allowReselect: true,
45546                         selectHandler: function(cp, color){
45547                             if(Roo.isGecko){
45548                                 editorcore.execCmd('useCSS', false);
45549                                 editorcore.execCmd('hilitecolor', color);
45550                                 editorcore.execCmd('useCSS', true);
45551                                 editor.deferFocus();
45552                             }else{
45553                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45554                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45555                                 editor.deferFocus();
45556                             }
45557                         },
45558                         scope:editorcore,
45559                         clickEvent:'mousedown'
45560                     })
45561                 }
45562             );
45563         };
45564         // now add all the items...
45565         
45566
45567         if(!this.disable.alignments){
45568             tb.add(
45569                 '-',
45570                 btn('justifyleft'),
45571                 btn('justifycenter'),
45572                 btn('justifyright')
45573             );
45574         };
45575
45576         //if(!Roo.isSafari){
45577             if(!this.disable.links){
45578                 tb.add(
45579                     '-',
45580                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45581                 );
45582             };
45583
45584             if(!this.disable.lists){
45585                 tb.add(
45586                     '-',
45587                     btn('insertorderedlist'),
45588                     btn('insertunorderedlist')
45589                 );
45590             }
45591             if(!this.disable.sourceEdit){
45592                 tb.add(
45593                     '-',
45594                     btn('sourceedit', true, function(btn){
45595                         this.toggleSourceEdit(btn.pressed);
45596                     })
45597                 );
45598             }
45599         //}
45600         
45601         var smenu = { };
45602         // special menu.. - needs to be tidied up..
45603         if (!this.disable.special) {
45604             smenu = {
45605                 text: "&#169;",
45606                 cls: 'x-edit-none',
45607                 
45608                 menu : {
45609                     items : []
45610                 }
45611             };
45612             for (var i =0; i < this.specialChars.length; i++) {
45613                 smenu.menu.items.push({
45614                     
45615                     html: this.specialChars[i],
45616                     handler: function(a,b) {
45617                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
45618                         //editor.insertAtCursor(a.html);
45619                         
45620                     },
45621                     tabIndex:-1
45622                 });
45623             }
45624             
45625             
45626             tb.add(smenu);
45627             
45628             
45629         }
45630         
45631         var cmenu = { };
45632         if (!this.disable.cleanStyles) {
45633             cmenu = {
45634                 cls: 'x-btn-icon x-btn-clear',
45635                 
45636                 menu : {
45637                     items : []
45638                 }
45639             };
45640             for (var i =0; i < this.cleanStyles.length; i++) {
45641                 cmenu.menu.items.push({
45642                     actiontype : this.cleanStyles[i],
45643                     html: 'Remove ' + this.cleanStyles[i],
45644                     handler: function(a,b) {
45645 //                        Roo.log(a);
45646 //                        Roo.log(b);
45647                         var c = Roo.get(editorcore.doc.body);
45648                         c.select('[style]').each(function(s) {
45649                             s.dom.style.removeProperty(a.actiontype);
45650                         });
45651                         editorcore.syncValue();
45652                     },
45653                     tabIndex:-1
45654                 });
45655             }
45656              cmenu.menu.items.push({
45657                 actiontype : 'tablewidths',
45658                 html: 'Remove Table Widths',
45659                 handler: function(a,b) {
45660                     editorcore.cleanTableWidths();
45661                     editorcore.syncValue();
45662                 },
45663                 tabIndex:-1
45664             });
45665             cmenu.menu.items.push({
45666                 actiontype : 'word',
45667                 html: 'Remove MS Word Formating',
45668                 handler: function(a,b) {
45669                     editorcore.cleanWord();
45670                     editorcore.syncValue();
45671                 },
45672                 tabIndex:-1
45673             });
45674             
45675             cmenu.menu.items.push({
45676                 actiontype : 'all',
45677                 html: 'Remove All Styles',
45678                 handler: function(a,b) {
45679                     
45680                     var c = Roo.get(editorcore.doc.body);
45681                     c.select('[style]').each(function(s) {
45682                         s.dom.removeAttribute('style');
45683                     });
45684                     editorcore.syncValue();
45685                 },
45686                 tabIndex:-1
45687             });
45688             
45689             cmenu.menu.items.push({
45690                 actiontype : 'all',
45691                 html: 'Remove All CSS Classes',
45692                 handler: function(a,b) {
45693                     
45694                     var c = Roo.get(editorcore.doc.body);
45695                     c.select('[class]').each(function(s) {
45696                         s.dom.removeAttribute('class');
45697                     });
45698                     editorcore.cleanWord();
45699                     editorcore.syncValue();
45700                 },
45701                 tabIndex:-1
45702             });
45703             
45704              cmenu.menu.items.push({
45705                 actiontype : 'tidy',
45706                 html: 'Tidy HTML Source',
45707                 handler: function(a,b) {
45708                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
45709                     editorcore.syncValue();
45710                 },
45711                 tabIndex:-1
45712             });
45713             
45714             
45715             tb.add(cmenu);
45716         }
45717          
45718         if (!this.disable.specialElements) {
45719             var semenu = {
45720                 text: "Other;",
45721                 cls: 'x-edit-none',
45722                 menu : {
45723                     items : []
45724                 }
45725             };
45726             for (var i =0; i < this.specialElements.length; i++) {
45727                 semenu.menu.items.push(
45728                     Roo.apply({ 
45729                         handler: function(a,b) {
45730                             editor.insertAtCursor(this.ihtml);
45731                         }
45732                     }, this.specialElements[i])
45733                 );
45734                     
45735             }
45736             
45737             tb.add(semenu);
45738             
45739             
45740         }
45741          
45742         
45743         if (this.btns) {
45744             for(var i =0; i< this.btns.length;i++) {
45745                 var b = Roo.factory(this.btns[i],Roo.form);
45746                 b.cls =  'x-edit-none';
45747                 
45748                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
45749                     b.cls += ' x-init-enable';
45750                 }
45751                 
45752                 b.scope = editorcore;
45753                 tb.add(b);
45754             }
45755         
45756         }
45757         
45758         
45759         
45760         // disable everything...
45761         
45762         this.tb.items.each(function(item){
45763             
45764            if(
45765                 item.id != editorcore.frameId+ '-sourceedit' && 
45766                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
45767             ){
45768                 
45769                 item.disable();
45770             }
45771         });
45772         this.rendered = true;
45773         
45774         // the all the btns;
45775         editor.on('editorevent', this.updateToolbar, this);
45776         // other toolbars need to implement this..
45777         //editor.on('editmodechange', this.updateToolbar, this);
45778     },
45779     
45780     
45781     relayBtnCmd : function(btn) {
45782         this.editorcore.relayCmd(btn.cmd);
45783     },
45784     // private used internally
45785     createLink : function(){
45786         Roo.log("create link?");
45787         var url = prompt(this.createLinkText, this.defaultLinkValue);
45788         if(url && url != 'http:/'+'/'){
45789             this.editorcore.relayCmd('createlink', url);
45790         }
45791     },
45792
45793     
45794     /**
45795      * Protected method that will not generally be called directly. It triggers
45796      * a toolbar update by reading the markup state of the current selection in the editor.
45797      */
45798     updateToolbar: function(){
45799
45800         if(!this.editorcore.activated){
45801             this.editor.onFirstFocus();
45802             return;
45803         }
45804
45805         var btns = this.tb.items.map, 
45806             doc = this.editorcore.doc,
45807             frameId = this.editorcore.frameId;
45808
45809         if(!this.disable.font && !Roo.isSafari){
45810             /*
45811             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
45812             if(name != this.fontSelect.dom.value){
45813                 this.fontSelect.dom.value = name;
45814             }
45815             */
45816         }
45817         if(!this.disable.format){
45818             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
45819             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
45820             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
45821             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
45822         }
45823         if(!this.disable.alignments){
45824             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
45825             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
45826             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
45827         }
45828         if(!Roo.isSafari && !this.disable.lists){
45829             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
45830             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
45831         }
45832         
45833         var ans = this.editorcore.getAllAncestors();
45834         if (this.formatCombo) {
45835             
45836             
45837             var store = this.formatCombo.store;
45838             this.formatCombo.setValue("");
45839             for (var i =0; i < ans.length;i++) {
45840                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
45841                     // select it..
45842                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
45843                     break;
45844                 }
45845             }
45846         }
45847         
45848         
45849         
45850         // hides menus... - so this cant be on a menu...
45851         Roo.menu.MenuMgr.hideAll();
45852
45853         //this.editorsyncValue();
45854     },
45855    
45856     
45857     createFontOptions : function(){
45858         var buf = [], fs = this.fontFamilies, ff, lc;
45859         
45860         
45861         
45862         for(var i = 0, len = fs.length; i< len; i++){
45863             ff = fs[i];
45864             lc = ff.toLowerCase();
45865             buf.push(
45866                 '<option value="',lc,'" style="font-family:',ff,';"',
45867                     (this.defaultFont == lc ? ' selected="true">' : '>'),
45868                     ff,
45869                 '</option>'
45870             );
45871         }
45872         return buf.join('');
45873     },
45874     
45875     toggleSourceEdit : function(sourceEditMode){
45876         
45877         Roo.log("toolbar toogle");
45878         if(sourceEditMode === undefined){
45879             sourceEditMode = !this.sourceEditMode;
45880         }
45881         this.sourceEditMode = sourceEditMode === true;
45882         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
45883         // just toggle the button?
45884         if(btn.pressed !== this.sourceEditMode){
45885             btn.toggle(this.sourceEditMode);
45886             return;
45887         }
45888         
45889         if(sourceEditMode){
45890             Roo.log("disabling buttons");
45891             this.tb.items.each(function(item){
45892                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
45893                     item.disable();
45894                 }
45895             });
45896           
45897         }else{
45898             Roo.log("enabling buttons");
45899             if(this.editorcore.initialized){
45900                 this.tb.items.each(function(item){
45901                     item.enable();
45902                 });
45903             }
45904             
45905         }
45906         Roo.log("calling toggole on editor");
45907         // tell the editor that it's been pressed..
45908         this.editor.toggleSourceEdit(sourceEditMode);
45909        
45910     },
45911      /**
45912      * Object collection of toolbar tooltips for the buttons in the editor. The key
45913      * is the command id associated with that button and the value is a valid QuickTips object.
45914      * For example:
45915 <pre><code>
45916 {
45917     bold : {
45918         title: 'Bold (Ctrl+B)',
45919         text: 'Make the selected text bold.',
45920         cls: 'x-html-editor-tip'
45921     },
45922     italic : {
45923         title: 'Italic (Ctrl+I)',
45924         text: 'Make the selected text italic.',
45925         cls: 'x-html-editor-tip'
45926     },
45927     ...
45928 </code></pre>
45929     * @type Object
45930      */
45931     buttonTips : {
45932         bold : {
45933             title: 'Bold (Ctrl+B)',
45934             text: 'Make the selected text bold.',
45935             cls: 'x-html-editor-tip'
45936         },
45937         italic : {
45938             title: 'Italic (Ctrl+I)',
45939             text: 'Make the selected text italic.',
45940             cls: 'x-html-editor-tip'
45941         },
45942         underline : {
45943             title: 'Underline (Ctrl+U)',
45944             text: 'Underline the selected text.',
45945             cls: 'x-html-editor-tip'
45946         },
45947         strikethrough : {
45948             title: 'Strikethrough',
45949             text: 'Strikethrough the selected text.',
45950             cls: 'x-html-editor-tip'
45951         },
45952         increasefontsize : {
45953             title: 'Grow Text',
45954             text: 'Increase the font size.',
45955             cls: 'x-html-editor-tip'
45956         },
45957         decreasefontsize : {
45958             title: 'Shrink Text',
45959             text: 'Decrease the font size.',
45960             cls: 'x-html-editor-tip'
45961         },
45962         backcolor : {
45963             title: 'Text Highlight Color',
45964             text: 'Change the background color of the selected text.',
45965             cls: 'x-html-editor-tip'
45966         },
45967         forecolor : {
45968             title: 'Font Color',
45969             text: 'Change the color of the selected text.',
45970             cls: 'x-html-editor-tip'
45971         },
45972         justifyleft : {
45973             title: 'Align Text Left',
45974             text: 'Align text to the left.',
45975             cls: 'x-html-editor-tip'
45976         },
45977         justifycenter : {
45978             title: 'Center Text',
45979             text: 'Center text in the editor.',
45980             cls: 'x-html-editor-tip'
45981         },
45982         justifyright : {
45983             title: 'Align Text Right',
45984             text: 'Align text to the right.',
45985             cls: 'x-html-editor-tip'
45986         },
45987         insertunorderedlist : {
45988             title: 'Bullet List',
45989             text: 'Start a bulleted list.',
45990             cls: 'x-html-editor-tip'
45991         },
45992         insertorderedlist : {
45993             title: 'Numbered List',
45994             text: 'Start a numbered list.',
45995             cls: 'x-html-editor-tip'
45996         },
45997         createlink : {
45998             title: 'Hyperlink',
45999             text: 'Make the selected text a hyperlink.',
46000             cls: 'x-html-editor-tip'
46001         },
46002         sourceedit : {
46003             title: 'Source Edit',
46004             text: 'Switch to source editing mode.',
46005             cls: 'x-html-editor-tip'
46006         }
46007     },
46008     // private
46009     onDestroy : function(){
46010         if(this.rendered){
46011             
46012             this.tb.items.each(function(item){
46013                 if(item.menu){
46014                     item.menu.removeAll();
46015                     if(item.menu.el){
46016                         item.menu.el.destroy();
46017                     }
46018                 }
46019                 item.destroy();
46020             });
46021              
46022         }
46023     },
46024     onFirstFocus: function() {
46025         this.tb.items.each(function(item){
46026            item.enable();
46027         });
46028     }
46029 });
46030
46031
46032
46033
46034 // <script type="text/javascript">
46035 /*
46036  * Based on
46037  * Ext JS Library 1.1.1
46038  * Copyright(c) 2006-2007, Ext JS, LLC.
46039  *  
46040  
46041  */
46042
46043  
46044 /**
46045  * @class Roo.form.HtmlEditor.ToolbarContext
46046  * Context Toolbar
46047  * 
46048  * Usage:
46049  *
46050  new Roo.form.HtmlEditor({
46051     ....
46052     toolbars : [
46053         { xtype: 'ToolbarStandard', styles : {} }
46054         { xtype: 'ToolbarContext', disable : {} }
46055     ]
46056 })
46057
46058      
46059  * 
46060  * @config : {Object} disable List of elements to disable.. (not done yet.)
46061  * @config : {Object} styles  Map of styles available.
46062  * 
46063  */
46064
46065 Roo.form.HtmlEditor.ToolbarContext = function(config)
46066 {
46067     
46068     Roo.apply(this, config);
46069     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46070     // dont call parent... till later.
46071     this.styles = this.styles || {};
46072 }
46073
46074  
46075
46076 Roo.form.HtmlEditor.ToolbarContext.types = {
46077     'IMG' : {
46078         width : {
46079             title: "Width",
46080             width: 40
46081         },
46082         height:  {
46083             title: "Height",
46084             width: 40
46085         },
46086         align: {
46087             title: "Align",
46088             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46089             width : 80
46090             
46091         },
46092         border: {
46093             title: "Border",
46094             width: 40
46095         },
46096         alt: {
46097             title: "Alt",
46098             width: 120
46099         },
46100         src : {
46101             title: "Src",
46102             width: 220
46103         }
46104         
46105     },
46106     'A' : {
46107         name : {
46108             title: "Name",
46109             width: 50
46110         },
46111         target:  {
46112             title: "Target",
46113             width: 120
46114         },
46115         href:  {
46116             title: "Href",
46117             width: 220
46118         } // border?
46119         
46120     },
46121     'TABLE' : {
46122         rows : {
46123             title: "Rows",
46124             width: 20
46125         },
46126         cols : {
46127             title: "Cols",
46128             width: 20
46129         },
46130         width : {
46131             title: "Width",
46132             width: 40
46133         },
46134         height : {
46135             title: "Height",
46136             width: 40
46137         },
46138         border : {
46139             title: "Border",
46140             width: 20
46141         }
46142     },
46143     'TD' : {
46144         width : {
46145             title: "Width",
46146             width: 40
46147         },
46148         height : {
46149             title: "Height",
46150             width: 40
46151         },   
46152         align: {
46153             title: "Align",
46154             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46155             width: 80
46156         },
46157         valign: {
46158             title: "Valign",
46159             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46160             width: 80
46161         },
46162         colspan: {
46163             title: "Colspan",
46164             width: 20
46165             
46166         },
46167          'font-family'  : {
46168             title : "Font",
46169             style : 'fontFamily',
46170             displayField: 'display',
46171             optname : 'font-family',
46172             width: 140
46173         }
46174     },
46175     'INPUT' : {
46176         name : {
46177             title: "name",
46178             width: 120
46179         },
46180         value : {
46181             title: "Value",
46182             width: 120
46183         },
46184         width : {
46185             title: "Width",
46186             width: 40
46187         }
46188     },
46189     'LABEL' : {
46190         'for' : {
46191             title: "For",
46192             width: 120
46193         }
46194     },
46195     'TEXTAREA' : {
46196           name : {
46197             title: "name",
46198             width: 120
46199         },
46200         rows : {
46201             title: "Rows",
46202             width: 20
46203         },
46204         cols : {
46205             title: "Cols",
46206             width: 20
46207         }
46208     },
46209     'SELECT' : {
46210         name : {
46211             title: "name",
46212             width: 120
46213         },
46214         selectoptions : {
46215             title: "Options",
46216             width: 200
46217         }
46218     },
46219     
46220     // should we really allow this??
46221     // should this just be 
46222     'BODY' : {
46223         title : {
46224             title: "Title",
46225             width: 200,
46226             disabled : true
46227         }
46228     },
46229     'SPAN' : {
46230         'font-family'  : {
46231             title : "Font",
46232             style : 'fontFamily',
46233             displayField: 'display',
46234             optname : 'font-family',
46235             width: 140
46236         }
46237     },
46238     'DIV' : {
46239         'font-family'  : {
46240             title : "Font",
46241             style : 'fontFamily',
46242             displayField: 'display',
46243             optname : 'font-family',
46244             width: 140
46245         }
46246     },
46247      'P' : {
46248         'font-family'  : {
46249             title : "Font",
46250             style : 'fontFamily',
46251             displayField: 'display',
46252             optname : 'font-family',
46253             width: 140
46254         }
46255     },
46256     
46257     '*' : {
46258         // empty..
46259     }
46260
46261 };
46262
46263 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46264 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46265
46266 Roo.form.HtmlEditor.ToolbarContext.options = {
46267         'font-family'  : [ 
46268                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46269                 [ 'Courier New', 'Courier New'],
46270                 [ 'Tahoma', 'Tahoma'],
46271                 [ 'Times New Roman,serif', 'Times'],
46272                 [ 'Verdana','Verdana' ]
46273         ]
46274 };
46275
46276 // fixme - these need to be configurable..
46277  
46278
46279 //Roo.form.HtmlEditor.ToolbarContext.types
46280
46281
46282 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46283     
46284     tb: false,
46285     
46286     rendered: false,
46287     
46288     editor : false,
46289     editorcore : false,
46290     /**
46291      * @cfg {Object} disable  List of toolbar elements to disable
46292          
46293      */
46294     disable : false,
46295     /**
46296      * @cfg {Object} styles List of styles 
46297      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46298      *
46299      * These must be defined in the page, so they get rendered correctly..
46300      * .headline { }
46301      * TD.underline { }
46302      * 
46303      */
46304     styles : false,
46305     
46306     options: false,
46307     
46308     toolbars : false,
46309     
46310     init : function(editor)
46311     {
46312         this.editor = editor;
46313         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46314         var editorcore = this.editorcore;
46315         
46316         var fid = editorcore.frameId;
46317         var etb = this;
46318         function btn(id, toggle, handler){
46319             var xid = fid + '-'+ id ;
46320             return {
46321                 id : xid,
46322                 cmd : id,
46323                 cls : 'x-btn-icon x-edit-'+id,
46324                 enableToggle:toggle !== false,
46325                 scope: editorcore, // was editor...
46326                 handler:handler||editorcore.relayBtnCmd,
46327                 clickEvent:'mousedown',
46328                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46329                 tabIndex:-1
46330             };
46331         }
46332         // create a new element.
46333         var wdiv = editor.wrap.createChild({
46334                 tag: 'div'
46335             }, editor.wrap.dom.firstChild.nextSibling, true);
46336         
46337         // can we do this more than once??
46338         
46339          // stop form submits
46340       
46341  
46342         // disable everything...
46343         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46344         this.toolbars = {};
46345            
46346         for (var i in  ty) {
46347           
46348             this.toolbars[i] = this.buildToolbar(ty[i],i);
46349         }
46350         this.tb = this.toolbars.BODY;
46351         this.tb.el.show();
46352         this.buildFooter();
46353         this.footer.show();
46354         editor.on('hide', function( ) { this.footer.hide() }, this);
46355         editor.on('show', function( ) { this.footer.show() }, this);
46356         
46357          
46358         this.rendered = true;
46359         
46360         // the all the btns;
46361         editor.on('editorevent', this.updateToolbar, this);
46362         // other toolbars need to implement this..
46363         //editor.on('editmodechange', this.updateToolbar, this);
46364     },
46365     
46366     
46367     
46368     /**
46369      * Protected method that will not generally be called directly. It triggers
46370      * a toolbar update by reading the markup state of the current selection in the editor.
46371      *
46372      * Note you can force an update by calling on('editorevent', scope, false)
46373      */
46374     updateToolbar: function(editor,ev,sel){
46375
46376         //Roo.log(ev);
46377         // capture mouse up - this is handy for selecting images..
46378         // perhaps should go somewhere else...
46379         if(!this.editorcore.activated){
46380              this.editor.onFirstFocus();
46381             return;
46382         }
46383         
46384         
46385         
46386         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46387         // selectNode - might want to handle IE?
46388         if (ev &&
46389             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46390             ev.target && ev.target.tagName == 'IMG') {
46391             // they have click on an image...
46392             // let's see if we can change the selection...
46393             sel = ev.target;
46394          
46395               var nodeRange = sel.ownerDocument.createRange();
46396             try {
46397                 nodeRange.selectNode(sel);
46398             } catch (e) {
46399                 nodeRange.selectNodeContents(sel);
46400             }
46401             //nodeRange.collapse(true);
46402             var s = this.editorcore.win.getSelection();
46403             s.removeAllRanges();
46404             s.addRange(nodeRange);
46405         }  
46406         
46407       
46408         var updateFooter = sel ? false : true;
46409         
46410         
46411         var ans = this.editorcore.getAllAncestors();
46412         
46413         // pick
46414         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46415         
46416         if (!sel) { 
46417             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46418             sel = sel ? sel : this.editorcore.doc.body;
46419             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46420             
46421         }
46422         // pick a menu that exists..
46423         var tn = sel.tagName.toUpperCase();
46424         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46425         
46426         tn = sel.tagName.toUpperCase();
46427         
46428         var lastSel = this.tb.selectedNode;
46429         
46430         this.tb.selectedNode = sel;
46431         
46432         // if current menu does not match..
46433         
46434         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46435                 
46436             this.tb.el.hide();
46437             ///console.log("show: " + tn);
46438             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46439             this.tb.el.show();
46440             // update name
46441             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46442             
46443             
46444             // update attributes
46445             if (this.tb.fields) {
46446                 this.tb.fields.each(function(e) {
46447                     if (e.stylename) {
46448                         e.setValue(sel.style[e.stylename]);
46449                         return;
46450                     } 
46451                    e.setValue(sel.getAttribute(e.attrname));
46452                 });
46453             }
46454             
46455             var hasStyles = false;
46456             for(var i in this.styles) {
46457                 hasStyles = true;
46458                 break;
46459             }
46460             
46461             // update styles
46462             if (hasStyles) { 
46463                 var st = this.tb.fields.item(0);
46464                 
46465                 st.store.removeAll();
46466                
46467                 
46468                 var cn = sel.className.split(/\s+/);
46469                 
46470                 var avs = [];
46471                 if (this.styles['*']) {
46472                     
46473                     Roo.each(this.styles['*'], function(v) {
46474                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46475                     });
46476                 }
46477                 if (this.styles[tn]) { 
46478                     Roo.each(this.styles[tn], function(v) {
46479                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46480                     });
46481                 }
46482                 
46483                 st.store.loadData(avs);
46484                 st.collapse();
46485                 st.setValue(cn);
46486             }
46487             // flag our selected Node.
46488             this.tb.selectedNode = sel;
46489            
46490            
46491             Roo.menu.MenuMgr.hideAll();
46492
46493         }
46494         
46495         if (!updateFooter) {
46496             //this.footDisp.dom.innerHTML = ''; 
46497             return;
46498         }
46499         // update the footer
46500         //
46501         var html = '';
46502         
46503         this.footerEls = ans.reverse();
46504         Roo.each(this.footerEls, function(a,i) {
46505             if (!a) { return; }
46506             html += html.length ? ' &gt; '  :  '';
46507             
46508             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46509             
46510         });
46511        
46512         // 
46513         var sz = this.footDisp.up('td').getSize();
46514         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46515         this.footDisp.dom.style.marginLeft = '5px';
46516         
46517         this.footDisp.dom.style.overflow = 'hidden';
46518         
46519         this.footDisp.dom.innerHTML = html;
46520             
46521         //this.editorsyncValue();
46522     },
46523      
46524     
46525    
46526        
46527     // private
46528     onDestroy : function(){
46529         if(this.rendered){
46530             
46531             this.tb.items.each(function(item){
46532                 if(item.menu){
46533                     item.menu.removeAll();
46534                     if(item.menu.el){
46535                         item.menu.el.destroy();
46536                     }
46537                 }
46538                 item.destroy();
46539             });
46540              
46541         }
46542     },
46543     onFirstFocus: function() {
46544         // need to do this for all the toolbars..
46545         this.tb.items.each(function(item){
46546            item.enable();
46547         });
46548     },
46549     buildToolbar: function(tlist, nm)
46550     {
46551         var editor = this.editor;
46552         var editorcore = this.editorcore;
46553          // create a new element.
46554         var wdiv = editor.wrap.createChild({
46555                 tag: 'div'
46556             }, editor.wrap.dom.firstChild.nextSibling, true);
46557         
46558        
46559         var tb = new Roo.Toolbar(wdiv);
46560         // add the name..
46561         
46562         tb.add(nm+ ":&nbsp;");
46563         
46564         var styles = [];
46565         for(var i in this.styles) {
46566             styles.push(i);
46567         }
46568         
46569         // styles...
46570         if (styles && styles.length) {
46571             
46572             // this needs a multi-select checkbox...
46573             tb.addField( new Roo.form.ComboBox({
46574                 store: new Roo.data.SimpleStore({
46575                     id : 'val',
46576                     fields: ['val', 'selected'],
46577                     data : [] 
46578                 }),
46579                 name : '-roo-edit-className',
46580                 attrname : 'className',
46581                 displayField: 'val',
46582                 typeAhead: false,
46583                 mode: 'local',
46584                 editable : false,
46585                 triggerAction: 'all',
46586                 emptyText:'Select Style',
46587                 selectOnFocus:true,
46588                 width: 130,
46589                 listeners : {
46590                     'select': function(c, r, i) {
46591                         // initial support only for on class per el..
46592                         tb.selectedNode.className =  r ? r.get('val') : '';
46593                         editorcore.syncValue();
46594                     }
46595                 }
46596     
46597             }));
46598         }
46599         
46600         var tbc = Roo.form.HtmlEditor.ToolbarContext;
46601         var tbops = tbc.options;
46602         
46603         for (var i in tlist) {
46604             
46605             var item = tlist[i];
46606             tb.add(item.title + ":&nbsp;");
46607             
46608             
46609             //optname == used so you can configure the options available..
46610             var opts = item.opts ? item.opts : false;
46611             if (item.optname) {
46612                 opts = tbops[item.optname];
46613            
46614             }
46615             
46616             if (opts) {
46617                 // opts == pulldown..
46618                 tb.addField( new Roo.form.ComboBox({
46619                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
46620                         id : 'val',
46621                         fields: ['val', 'display'],
46622                         data : opts  
46623                     }),
46624                     name : '-roo-edit-' + i,
46625                     attrname : i,
46626                     stylename : item.style ? item.style : false,
46627                     displayField: item.displayField ? item.displayField : 'val',
46628                     valueField :  'val',
46629                     typeAhead: false,
46630                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
46631                     editable : false,
46632                     triggerAction: 'all',
46633                     emptyText:'Select',
46634                     selectOnFocus:true,
46635                     width: item.width ? item.width  : 130,
46636                     listeners : {
46637                         'select': function(c, r, i) {
46638                             if (c.stylename) {
46639                                 tb.selectedNode.style[c.stylename] =  r.get('val');
46640                                 return;
46641                             }
46642                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
46643                         }
46644                     }
46645
46646                 }));
46647                 continue;
46648                     
46649                  
46650                 
46651                 tb.addField( new Roo.form.TextField({
46652                     name: i,
46653                     width: 100,
46654                     //allowBlank:false,
46655                     value: ''
46656                 }));
46657                 continue;
46658             }
46659             tb.addField( new Roo.form.TextField({
46660                 name: '-roo-edit-' + i,
46661                 attrname : i,
46662                 
46663                 width: item.width,
46664                 //allowBlank:true,
46665                 value: '',
46666                 listeners: {
46667                     'change' : function(f, nv, ov) {
46668                         tb.selectedNode.setAttribute(f.attrname, nv);
46669                         editorcore.syncValue();
46670                     }
46671                 }
46672             }));
46673              
46674         }
46675         
46676         var _this = this;
46677         
46678         if(nm == 'BODY'){
46679             tb.addSeparator();
46680         
46681             tb.addButton( {
46682                 text: 'Stylesheets',
46683
46684                 listeners : {
46685                     click : function ()
46686                     {
46687                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
46688                     }
46689                 }
46690             });
46691         }
46692         
46693         tb.addFill();
46694         tb.addButton( {
46695             text: 'Remove Tag',
46696     
46697             listeners : {
46698                 click : function ()
46699                 {
46700                     // remove
46701                     // undo does not work.
46702                      
46703                     var sn = tb.selectedNode;
46704                     
46705                     var pn = sn.parentNode;
46706                     
46707                     var stn =  sn.childNodes[0];
46708                     var en = sn.childNodes[sn.childNodes.length - 1 ];
46709                     while (sn.childNodes.length) {
46710                         var node = sn.childNodes[0];
46711                         sn.removeChild(node);
46712                         //Roo.log(node);
46713                         pn.insertBefore(node, sn);
46714                         
46715                     }
46716                     pn.removeChild(sn);
46717                     var range = editorcore.createRange();
46718         
46719                     range.setStart(stn,0);
46720                     range.setEnd(en,0); //????
46721                     //range.selectNode(sel);
46722                     
46723                     
46724                     var selection = editorcore.getSelection();
46725                     selection.removeAllRanges();
46726                     selection.addRange(range);
46727                     
46728                     
46729                     
46730                     //_this.updateToolbar(null, null, pn);
46731                     _this.updateToolbar(null, null, null);
46732                     _this.footDisp.dom.innerHTML = ''; 
46733                 }
46734             }
46735             
46736                     
46737                 
46738             
46739         });
46740         
46741         
46742         tb.el.on('click', function(e){
46743             e.preventDefault(); // what does this do?
46744         });
46745         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
46746         tb.el.hide();
46747         tb.name = nm;
46748         // dont need to disable them... as they will get hidden
46749         return tb;
46750          
46751         
46752     },
46753     buildFooter : function()
46754     {
46755         
46756         var fel = this.editor.wrap.createChild();
46757         this.footer = new Roo.Toolbar(fel);
46758         // toolbar has scrolly on left / right?
46759         var footDisp= new Roo.Toolbar.Fill();
46760         var _t = this;
46761         this.footer.add(
46762             {
46763                 text : '&lt;',
46764                 xtype: 'Button',
46765                 handler : function() {
46766                     _t.footDisp.scrollTo('left',0,true)
46767                 }
46768             }
46769         );
46770         this.footer.add( footDisp );
46771         this.footer.add( 
46772             {
46773                 text : '&gt;',
46774                 xtype: 'Button',
46775                 handler : function() {
46776                     // no animation..
46777                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
46778                 }
46779             }
46780         );
46781         var fel = Roo.get(footDisp.el);
46782         fel.addClass('x-editor-context');
46783         this.footDispWrap = fel; 
46784         this.footDispWrap.overflow  = 'hidden';
46785         
46786         this.footDisp = fel.createChild();
46787         this.footDispWrap.on('click', this.onContextClick, this)
46788         
46789         
46790     },
46791     onContextClick : function (ev,dom)
46792     {
46793         ev.preventDefault();
46794         var  cn = dom.className;
46795         //Roo.log(cn);
46796         if (!cn.match(/x-ed-loc-/)) {
46797             return;
46798         }
46799         var n = cn.split('-').pop();
46800         var ans = this.footerEls;
46801         var sel = ans[n];
46802         
46803          // pick
46804         var range = this.editorcore.createRange();
46805         
46806         range.selectNodeContents(sel);
46807         //range.selectNode(sel);
46808         
46809         
46810         var selection = this.editorcore.getSelection();
46811         selection.removeAllRanges();
46812         selection.addRange(range);
46813         
46814         
46815         
46816         this.updateToolbar(null, null, sel);
46817         
46818         
46819     }
46820     
46821     
46822     
46823     
46824     
46825 });
46826
46827
46828
46829
46830
46831 /*
46832  * Based on:
46833  * Ext JS Library 1.1.1
46834  * Copyright(c) 2006-2007, Ext JS, LLC.
46835  *
46836  * Originally Released Under LGPL - original licence link has changed is not relivant.
46837  *
46838  * Fork - LGPL
46839  * <script type="text/javascript">
46840  */
46841  
46842 /**
46843  * @class Roo.form.BasicForm
46844  * @extends Roo.util.Observable
46845  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
46846  * @constructor
46847  * @param {String/HTMLElement/Roo.Element} el The form element or its id
46848  * @param {Object} config Configuration options
46849  */
46850 Roo.form.BasicForm = function(el, config){
46851     this.allItems = [];
46852     this.childForms = [];
46853     Roo.apply(this, config);
46854     /*
46855      * The Roo.form.Field items in this form.
46856      * @type MixedCollection
46857      */
46858      
46859      
46860     this.items = new Roo.util.MixedCollection(false, function(o){
46861         return o.id || (o.id = Roo.id());
46862     });
46863     this.addEvents({
46864         /**
46865          * @event beforeaction
46866          * Fires before any action is performed. Return false to cancel the action.
46867          * @param {Form} this
46868          * @param {Action} action The action to be performed
46869          */
46870         beforeaction: true,
46871         /**
46872          * @event actionfailed
46873          * Fires when an action fails.
46874          * @param {Form} this
46875          * @param {Action} action The action that failed
46876          */
46877         actionfailed : true,
46878         /**
46879          * @event actioncomplete
46880          * Fires when an action is completed.
46881          * @param {Form} this
46882          * @param {Action} action The action that completed
46883          */
46884         actioncomplete : true
46885     });
46886     if(el){
46887         this.initEl(el);
46888     }
46889     Roo.form.BasicForm.superclass.constructor.call(this);
46890     
46891     Roo.form.BasicForm.popover.apply();
46892 };
46893
46894 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
46895     /**
46896      * @cfg {String} method
46897      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
46898      */
46899     /**
46900      * @cfg {DataReader} reader
46901      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
46902      * This is optional as there is built-in support for processing JSON.
46903      */
46904     /**
46905      * @cfg {DataReader} errorReader
46906      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
46907      * This is completely optional as there is built-in support for processing JSON.
46908      */
46909     /**
46910      * @cfg {String} url
46911      * The URL to use for form actions if one isn't supplied in the action options.
46912      */
46913     /**
46914      * @cfg {Boolean} fileUpload
46915      * Set to true if this form is a file upload.
46916      */
46917      
46918     /**
46919      * @cfg {Object} baseParams
46920      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
46921      */
46922      /**
46923      
46924     /**
46925      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
46926      */
46927     timeout: 30,
46928
46929     // private
46930     activeAction : null,
46931
46932     /**
46933      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
46934      * or setValues() data instead of when the form was first created.
46935      */
46936     trackResetOnLoad : false,
46937     
46938     
46939     /**
46940      * childForms - used for multi-tab forms
46941      * @type {Array}
46942      */
46943     childForms : false,
46944     
46945     /**
46946      * allItems - full list of fields.
46947      * @type {Array}
46948      */
46949     allItems : false,
46950     
46951     /**
46952      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
46953      * element by passing it or its id or mask the form itself by passing in true.
46954      * @type Mixed
46955      */
46956     waitMsgTarget : false,
46957     
46958     /**
46959      * @type Boolean
46960      */
46961     disableMask : false,
46962     
46963     /**
46964      * @cfg {Boolean} errorMask (true|false) default false
46965      */
46966     errorMask : false,
46967     
46968     /**
46969      * @cfg {Number} maskOffset Default 100
46970      */
46971     maskOffset : 100,
46972
46973     // private
46974     initEl : function(el){
46975         this.el = Roo.get(el);
46976         this.id = this.el.id || Roo.id();
46977         this.el.on('submit', this.onSubmit, this);
46978         this.el.addClass('x-form');
46979     },
46980
46981     // private
46982     onSubmit : function(e){
46983         e.stopEvent();
46984     },
46985
46986     /**
46987      * Returns true if client-side validation on the form is successful.
46988      * @return Boolean
46989      */
46990     isValid : function(){
46991         var valid = true;
46992         var target = false;
46993         this.items.each(function(f){
46994             if(f.validate()){
46995                 return;
46996             }
46997             
46998             valid = false;
46999                 
47000             if(!target && f.el.isVisible(true)){
47001                 target = f;
47002             }
47003         });
47004         
47005         if(this.errorMask && !valid){
47006             Roo.form.BasicForm.popover.mask(this, target);
47007         }
47008         
47009         return valid;
47010     },
47011
47012     /**
47013      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47014      * @return Boolean
47015      */
47016     isDirty : function(){
47017         var dirty = false;
47018         this.items.each(function(f){
47019            if(f.isDirty()){
47020                dirty = true;
47021                return false;
47022            }
47023         });
47024         return dirty;
47025     },
47026     
47027     /**
47028      * Returns true if any fields in this form have changed since their original load. (New version)
47029      * @return Boolean
47030      */
47031     
47032     hasChanged : function()
47033     {
47034         var dirty = false;
47035         this.items.each(function(f){
47036            if(f.hasChanged()){
47037                dirty = true;
47038                return false;
47039            }
47040         });
47041         return dirty;
47042         
47043     },
47044     /**
47045      * Resets all hasChanged to 'false' -
47046      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47047      * So hasChanged storage is only to be used for this purpose
47048      * @return Boolean
47049      */
47050     resetHasChanged : function()
47051     {
47052         this.items.each(function(f){
47053            f.resetHasChanged();
47054         });
47055         
47056     },
47057     
47058     
47059     /**
47060      * Performs a predefined action (submit or load) or custom actions you define on this form.
47061      * @param {String} actionName The name of the action type
47062      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47063      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47064      * accept other config options):
47065      * <pre>
47066 Property          Type             Description
47067 ----------------  ---------------  ----------------------------------------------------------------------------------
47068 url               String           The url for the action (defaults to the form's url)
47069 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47070 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47071 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47072                                    validate the form on the client (defaults to false)
47073      * </pre>
47074      * @return {BasicForm} this
47075      */
47076     doAction : function(action, options){
47077         if(typeof action == 'string'){
47078             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47079         }
47080         if(this.fireEvent('beforeaction', this, action) !== false){
47081             this.beforeAction(action);
47082             action.run.defer(100, action);
47083         }
47084         return this;
47085     },
47086
47087     /**
47088      * Shortcut to do a submit action.
47089      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47090      * @return {BasicForm} this
47091      */
47092     submit : function(options){
47093         this.doAction('submit', options);
47094         return this;
47095     },
47096
47097     /**
47098      * Shortcut to do a load action.
47099      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47100      * @return {BasicForm} this
47101      */
47102     load : function(options){
47103         this.doAction('load', options);
47104         return this;
47105     },
47106
47107     /**
47108      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47109      * @param {Record} record The record to edit
47110      * @return {BasicForm} this
47111      */
47112     updateRecord : function(record){
47113         record.beginEdit();
47114         var fs = record.fields;
47115         fs.each(function(f){
47116             var field = this.findField(f.name);
47117             if(field){
47118                 record.set(f.name, field.getValue());
47119             }
47120         }, this);
47121         record.endEdit();
47122         return this;
47123     },
47124
47125     /**
47126      * Loads an Roo.data.Record into this form.
47127      * @param {Record} record The record to load
47128      * @return {BasicForm} this
47129      */
47130     loadRecord : function(record){
47131         this.setValues(record.data);
47132         return this;
47133     },
47134
47135     // private
47136     beforeAction : function(action){
47137         var o = action.options;
47138         
47139         if(!this.disableMask) {
47140             if(this.waitMsgTarget === true){
47141                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47142             }else if(this.waitMsgTarget){
47143                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47144                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47145             }else {
47146                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47147             }
47148         }
47149         
47150          
47151     },
47152
47153     // private
47154     afterAction : function(action, success){
47155         this.activeAction = null;
47156         var o = action.options;
47157         
47158         if(!this.disableMask) {
47159             if(this.waitMsgTarget === true){
47160                 this.el.unmask();
47161             }else if(this.waitMsgTarget){
47162                 this.waitMsgTarget.unmask();
47163             }else{
47164                 Roo.MessageBox.updateProgress(1);
47165                 Roo.MessageBox.hide();
47166             }
47167         }
47168         
47169         if(success){
47170             if(o.reset){
47171                 this.reset();
47172             }
47173             Roo.callback(o.success, o.scope, [this, action]);
47174             this.fireEvent('actioncomplete', this, action);
47175             
47176         }else{
47177             
47178             // failure condition..
47179             // we have a scenario where updates need confirming.
47180             // eg. if a locking scenario exists..
47181             // we look for { errors : { needs_confirm : true }} in the response.
47182             if (
47183                 (typeof(action.result) != 'undefined')  &&
47184                 (typeof(action.result.errors) != 'undefined')  &&
47185                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47186            ){
47187                 var _t = this;
47188                 Roo.MessageBox.confirm(
47189                     "Change requires confirmation",
47190                     action.result.errorMsg,
47191                     function(r) {
47192                         if (r != 'yes') {
47193                             return;
47194                         }
47195                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47196                     }
47197                     
47198                 );
47199                 
47200                 
47201                 
47202                 return;
47203             }
47204             
47205             Roo.callback(o.failure, o.scope, [this, action]);
47206             // show an error message if no failed handler is set..
47207             if (!this.hasListener('actionfailed')) {
47208                 Roo.MessageBox.alert("Error",
47209                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47210                         action.result.errorMsg :
47211                         "Saving Failed, please check your entries or try again"
47212                 );
47213             }
47214             
47215             this.fireEvent('actionfailed', this, action);
47216         }
47217         
47218     },
47219
47220     /**
47221      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47222      * @param {String} id The value to search for
47223      * @return Field
47224      */
47225     findField : function(id){
47226         var field = this.items.get(id);
47227         if(!field){
47228             this.items.each(function(f){
47229                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47230                     field = f;
47231                     return false;
47232                 }
47233             });
47234         }
47235         return field || null;
47236     },
47237
47238     /**
47239      * Add a secondary form to this one, 
47240      * Used to provide tabbed forms. One form is primary, with hidden values 
47241      * which mirror the elements from the other forms.
47242      * 
47243      * @param {Roo.form.Form} form to add.
47244      * 
47245      */
47246     addForm : function(form)
47247     {
47248        
47249         if (this.childForms.indexOf(form) > -1) {
47250             // already added..
47251             return;
47252         }
47253         this.childForms.push(form);
47254         var n = '';
47255         Roo.each(form.allItems, function (fe) {
47256             
47257             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47258             if (this.findField(n)) { // already added..
47259                 return;
47260             }
47261             var add = new Roo.form.Hidden({
47262                 name : n
47263             });
47264             add.render(this.el);
47265             
47266             this.add( add );
47267         }, this);
47268         
47269     },
47270     /**
47271      * Mark fields in this form invalid in bulk.
47272      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47273      * @return {BasicForm} this
47274      */
47275     markInvalid : function(errors){
47276         if(errors instanceof Array){
47277             for(var i = 0, len = errors.length; i < len; i++){
47278                 var fieldError = errors[i];
47279                 var f = this.findField(fieldError.id);
47280                 if(f){
47281                     f.markInvalid(fieldError.msg);
47282                 }
47283             }
47284         }else{
47285             var field, id;
47286             for(id in errors){
47287                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47288                     field.markInvalid(errors[id]);
47289                 }
47290             }
47291         }
47292         Roo.each(this.childForms || [], function (f) {
47293             f.markInvalid(errors);
47294         });
47295         
47296         return this;
47297     },
47298
47299     /**
47300      * Set values for fields in this form in bulk.
47301      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47302      * @return {BasicForm} this
47303      */
47304     setValues : function(values){
47305         if(values instanceof Array){ // array of objects
47306             for(var i = 0, len = values.length; i < len; i++){
47307                 var v = values[i];
47308                 var f = this.findField(v.id);
47309                 if(f){
47310                     f.setValue(v.value);
47311                     if(this.trackResetOnLoad){
47312                         f.originalValue = f.getValue();
47313                     }
47314                 }
47315             }
47316         }else{ // object hash
47317             var field, id;
47318             for(id in values){
47319                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47320                     
47321                     if (field.setFromData && 
47322                         field.valueField && 
47323                         field.displayField &&
47324                         // combos' with local stores can 
47325                         // be queried via setValue()
47326                         // to set their value..
47327                         (field.store && !field.store.isLocal)
47328                         ) {
47329                         // it's a combo
47330                         var sd = { };
47331                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47332                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47333                         field.setFromData(sd);
47334                         
47335                     } else {
47336                         field.setValue(values[id]);
47337                     }
47338                     
47339                     
47340                     if(this.trackResetOnLoad){
47341                         field.originalValue = field.getValue();
47342                     }
47343                 }
47344             }
47345         }
47346         this.resetHasChanged();
47347         
47348         
47349         Roo.each(this.childForms || [], function (f) {
47350             f.setValues(values);
47351             f.resetHasChanged();
47352         });
47353                 
47354         return this;
47355     },
47356  
47357     /**
47358      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47359      * they are returned as an array.
47360      * @param {Boolean} asString
47361      * @return {Object}
47362      */
47363     getValues : function(asString){
47364         if (this.childForms) {
47365             // copy values from the child forms
47366             Roo.each(this.childForms, function (f) {
47367                 this.setValues(f.getValues());
47368             }, this);
47369         }
47370         
47371         // use formdata
47372         if (typeof(FormData) != 'undefined' && asString !== true) {
47373             var fd = (new FormData(this.el.dom)).entries();
47374             var ret = {};
47375             var ent = fd.next();
47376             while (!ent.done) {
47377                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47378                 ent = fd.next();
47379             };
47380             return ret;
47381         }
47382         
47383         
47384         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47385         if(asString === true){
47386             return fs;
47387         }
47388         return Roo.urlDecode(fs);
47389     },
47390     
47391     /**
47392      * Returns the fields in this form as an object with key/value pairs. 
47393      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47394      * @return {Object}
47395      */
47396     getFieldValues : function(with_hidden)
47397     {
47398         if (this.childForms) {
47399             // copy values from the child forms
47400             // should this call getFieldValues - probably not as we do not currently copy
47401             // hidden fields when we generate..
47402             Roo.each(this.childForms, function (f) {
47403                 this.setValues(f.getValues());
47404             }, this);
47405         }
47406         
47407         var ret = {};
47408         this.items.each(function(f){
47409             if (!f.getName()) {
47410                 return;
47411             }
47412             var v = f.getValue();
47413             if (f.inputType =='radio') {
47414                 if (typeof(ret[f.getName()]) == 'undefined') {
47415                     ret[f.getName()] = ''; // empty..
47416                 }
47417                 
47418                 if (!f.el.dom.checked) {
47419                     return;
47420                     
47421                 }
47422                 v = f.el.dom.value;
47423                 
47424             }
47425             
47426             // not sure if this supported any more..
47427             if ((typeof(v) == 'object') && f.getRawValue) {
47428                 v = f.getRawValue() ; // dates..
47429             }
47430             // combo boxes where name != hiddenName...
47431             if (f.name != f.getName()) {
47432                 ret[f.name] = f.getRawValue();
47433             }
47434             ret[f.getName()] = v;
47435         });
47436         
47437         return ret;
47438     },
47439
47440     /**
47441      * Clears all invalid messages in this form.
47442      * @return {BasicForm} this
47443      */
47444     clearInvalid : function(){
47445         this.items.each(function(f){
47446            f.clearInvalid();
47447         });
47448         
47449         Roo.each(this.childForms || [], function (f) {
47450             f.clearInvalid();
47451         });
47452         
47453         
47454         return this;
47455     },
47456
47457     /**
47458      * Resets this form.
47459      * @return {BasicForm} this
47460      */
47461     reset : function(){
47462         this.items.each(function(f){
47463             f.reset();
47464         });
47465         
47466         Roo.each(this.childForms || [], function (f) {
47467             f.reset();
47468         });
47469         this.resetHasChanged();
47470         
47471         return this;
47472     },
47473
47474     /**
47475      * Add Roo.form components to this form.
47476      * @param {Field} field1
47477      * @param {Field} field2 (optional)
47478      * @param {Field} etc (optional)
47479      * @return {BasicForm} this
47480      */
47481     add : function(){
47482         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47483         return this;
47484     },
47485
47486
47487     /**
47488      * Removes a field from the items collection (does NOT remove its markup).
47489      * @param {Field} field
47490      * @return {BasicForm} this
47491      */
47492     remove : function(field){
47493         this.items.remove(field);
47494         return this;
47495     },
47496
47497     /**
47498      * Looks at the fields in this form, checks them for an id attribute,
47499      * and calls applyTo on the existing dom element with that id.
47500      * @return {BasicForm} this
47501      */
47502     render : function(){
47503         this.items.each(function(f){
47504             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47505                 f.applyTo(f.id);
47506             }
47507         });
47508         return this;
47509     },
47510
47511     /**
47512      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47513      * @param {Object} values
47514      * @return {BasicForm} this
47515      */
47516     applyToFields : function(o){
47517         this.items.each(function(f){
47518            Roo.apply(f, o);
47519         });
47520         return this;
47521     },
47522
47523     /**
47524      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47525      * @param {Object} values
47526      * @return {BasicForm} this
47527      */
47528     applyIfToFields : function(o){
47529         this.items.each(function(f){
47530            Roo.applyIf(f, o);
47531         });
47532         return this;
47533     }
47534 });
47535
47536 // back compat
47537 Roo.BasicForm = Roo.form.BasicForm;
47538
47539 Roo.apply(Roo.form.BasicForm, {
47540     
47541     popover : {
47542         
47543         padding : 5,
47544         
47545         isApplied : false,
47546         
47547         isMasked : false,
47548         
47549         form : false,
47550         
47551         target : false,
47552         
47553         intervalID : false,
47554         
47555         maskEl : false,
47556         
47557         apply : function()
47558         {
47559             if(this.isApplied){
47560                 return;
47561             }
47562             
47563             this.maskEl = {
47564                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
47565                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
47566                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
47567                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
47568             };
47569             
47570             this.maskEl.top.enableDisplayMode("block");
47571             this.maskEl.left.enableDisplayMode("block");
47572             this.maskEl.bottom.enableDisplayMode("block");
47573             this.maskEl.right.enableDisplayMode("block");
47574             
47575             Roo.get(document.body).on('click', function(){
47576                 this.unmask();
47577             }, this);
47578             
47579             Roo.get(document.body).on('touchstart', function(){
47580                 this.unmask();
47581             }, this);
47582             
47583             this.isApplied = true
47584         },
47585         
47586         mask : function(form, target)
47587         {
47588             this.form = form;
47589             
47590             this.target = target;
47591             
47592             if(!this.form.errorMask || !target.el){
47593                 return;
47594             }
47595             
47596             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
47597             
47598             var ot = this.target.el.calcOffsetsTo(scrollable);
47599             
47600             var scrollTo = ot[1] - this.form.maskOffset;
47601             
47602             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
47603             
47604             scrollable.scrollTo('top', scrollTo);
47605             
47606             var el = this.target.wrap || this.target.el;
47607             
47608             var box = el.getBox();
47609             
47610             this.maskEl.top.setStyle('position', 'absolute');
47611             this.maskEl.top.setStyle('z-index', 10000);
47612             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
47613             this.maskEl.top.setLeft(0);
47614             this.maskEl.top.setTop(0);
47615             this.maskEl.top.show();
47616             
47617             this.maskEl.left.setStyle('position', 'absolute');
47618             this.maskEl.left.setStyle('z-index', 10000);
47619             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
47620             this.maskEl.left.setLeft(0);
47621             this.maskEl.left.setTop(box.y - this.padding);
47622             this.maskEl.left.show();
47623
47624             this.maskEl.bottom.setStyle('position', 'absolute');
47625             this.maskEl.bottom.setStyle('z-index', 10000);
47626             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
47627             this.maskEl.bottom.setLeft(0);
47628             this.maskEl.bottom.setTop(box.bottom + this.padding);
47629             this.maskEl.bottom.show();
47630
47631             this.maskEl.right.setStyle('position', 'absolute');
47632             this.maskEl.right.setStyle('z-index', 10000);
47633             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
47634             this.maskEl.right.setLeft(box.right + this.padding);
47635             this.maskEl.right.setTop(box.y - this.padding);
47636             this.maskEl.right.show();
47637
47638             this.intervalID = window.setInterval(function() {
47639                 Roo.form.BasicForm.popover.unmask();
47640             }, 10000);
47641
47642             window.onwheel = function(){ return false;};
47643             
47644             (function(){ this.isMasked = true; }).defer(500, this);
47645             
47646         },
47647         
47648         unmask : function()
47649         {
47650             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
47651                 return;
47652             }
47653             
47654             this.maskEl.top.setStyle('position', 'absolute');
47655             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
47656             this.maskEl.top.hide();
47657
47658             this.maskEl.left.setStyle('position', 'absolute');
47659             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
47660             this.maskEl.left.hide();
47661
47662             this.maskEl.bottom.setStyle('position', 'absolute');
47663             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
47664             this.maskEl.bottom.hide();
47665
47666             this.maskEl.right.setStyle('position', 'absolute');
47667             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
47668             this.maskEl.right.hide();
47669             
47670             window.onwheel = function(){ return true;};
47671             
47672             if(this.intervalID){
47673                 window.clearInterval(this.intervalID);
47674                 this.intervalID = false;
47675             }
47676             
47677             this.isMasked = false;
47678             
47679         }
47680         
47681     }
47682     
47683 });/*
47684  * Based on:
47685  * Ext JS Library 1.1.1
47686  * Copyright(c) 2006-2007, Ext JS, LLC.
47687  *
47688  * Originally Released Under LGPL - original licence link has changed is not relivant.
47689  *
47690  * Fork - LGPL
47691  * <script type="text/javascript">
47692  */
47693
47694 /**
47695  * @class Roo.form.Form
47696  * @extends Roo.form.BasicForm
47697  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
47698  * @constructor
47699  * @param {Object} config Configuration options
47700  */
47701 Roo.form.Form = function(config){
47702     var xitems =  [];
47703     if (config.items) {
47704         xitems = config.items;
47705         delete config.items;
47706     }
47707    
47708     
47709     Roo.form.Form.superclass.constructor.call(this, null, config);
47710     this.url = this.url || this.action;
47711     if(!this.root){
47712         this.root = new Roo.form.Layout(Roo.applyIf({
47713             id: Roo.id()
47714         }, config));
47715     }
47716     this.active = this.root;
47717     /**
47718      * Array of all the buttons that have been added to this form via {@link addButton}
47719      * @type Array
47720      */
47721     this.buttons = [];
47722     this.allItems = [];
47723     this.addEvents({
47724         /**
47725          * @event clientvalidation
47726          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
47727          * @param {Form} this
47728          * @param {Boolean} valid true if the form has passed client-side validation
47729          */
47730         clientvalidation: true,
47731         /**
47732          * @event rendered
47733          * Fires when the form is rendered
47734          * @param {Roo.form.Form} form
47735          */
47736         rendered : true
47737     });
47738     
47739     if (this.progressUrl) {
47740             // push a hidden field onto the list of fields..
47741             this.addxtype( {
47742                     xns: Roo.form, 
47743                     xtype : 'Hidden', 
47744                     name : 'UPLOAD_IDENTIFIER' 
47745             });
47746         }
47747         
47748     
47749     Roo.each(xitems, this.addxtype, this);
47750     
47751 };
47752
47753 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
47754     /**
47755      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
47756      */
47757     /**
47758      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
47759      */
47760     /**
47761      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
47762      */
47763     buttonAlign:'center',
47764
47765     /**
47766      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
47767      */
47768     minButtonWidth:75,
47769
47770     /**
47771      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
47772      * This property cascades to child containers if not set.
47773      */
47774     labelAlign:'left',
47775
47776     /**
47777      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
47778      * fires a looping event with that state. This is required to bind buttons to the valid
47779      * state using the config value formBind:true on the button.
47780      */
47781     monitorValid : false,
47782
47783     /**
47784      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
47785      */
47786     monitorPoll : 200,
47787     
47788     /**
47789      * @cfg {String} progressUrl - Url to return progress data 
47790      */
47791     
47792     progressUrl : false,
47793     /**
47794      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
47795      * sending a formdata with extra parameters - eg uploaded elements.
47796      */
47797     
47798     formData : false,
47799     
47800     /**
47801      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
47802      * fields are added and the column is closed. If no fields are passed the column remains open
47803      * until end() is called.
47804      * @param {Object} config The config to pass to the column
47805      * @param {Field} field1 (optional)
47806      * @param {Field} field2 (optional)
47807      * @param {Field} etc (optional)
47808      * @return Column The column container object
47809      */
47810     column : function(c){
47811         var col = new Roo.form.Column(c);
47812         this.start(col);
47813         if(arguments.length > 1){ // duplicate code required because of Opera
47814             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47815             this.end();
47816         }
47817         return col;
47818     },
47819
47820     /**
47821      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
47822      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
47823      * until end() is called.
47824      * @param {Object} config The config to pass to the fieldset
47825      * @param {Field} field1 (optional)
47826      * @param {Field} field2 (optional)
47827      * @param {Field} etc (optional)
47828      * @return FieldSet The fieldset container object
47829      */
47830     fieldset : function(c){
47831         var fs = new Roo.form.FieldSet(c);
47832         this.start(fs);
47833         if(arguments.length > 1){ // duplicate code required because of Opera
47834             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47835             this.end();
47836         }
47837         return fs;
47838     },
47839
47840     /**
47841      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
47842      * fields are added and the container is closed. If no fields are passed the container remains open
47843      * until end() is called.
47844      * @param {Object} config The config to pass to the Layout
47845      * @param {Field} field1 (optional)
47846      * @param {Field} field2 (optional)
47847      * @param {Field} etc (optional)
47848      * @return Layout The container object
47849      */
47850     container : function(c){
47851         var l = new Roo.form.Layout(c);
47852         this.start(l);
47853         if(arguments.length > 1){ // duplicate code required because of Opera
47854             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47855             this.end();
47856         }
47857         return l;
47858     },
47859
47860     /**
47861      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
47862      * @param {Object} container A Roo.form.Layout or subclass of Layout
47863      * @return {Form} this
47864      */
47865     start : function(c){
47866         // cascade label info
47867         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
47868         this.active.stack.push(c);
47869         c.ownerCt = this.active;
47870         this.active = c;
47871         return this;
47872     },
47873
47874     /**
47875      * Closes the current open container
47876      * @return {Form} this
47877      */
47878     end : function(){
47879         if(this.active == this.root){
47880             return this;
47881         }
47882         this.active = this.active.ownerCt;
47883         return this;
47884     },
47885
47886     /**
47887      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
47888      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
47889      * as the label of the field.
47890      * @param {Field} field1
47891      * @param {Field} field2 (optional)
47892      * @param {Field} etc. (optional)
47893      * @return {Form} this
47894      */
47895     add : function(){
47896         this.active.stack.push.apply(this.active.stack, arguments);
47897         this.allItems.push.apply(this.allItems,arguments);
47898         var r = [];
47899         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
47900             if(a[i].isFormField){
47901                 r.push(a[i]);
47902             }
47903         }
47904         if(r.length > 0){
47905             Roo.form.Form.superclass.add.apply(this, r);
47906         }
47907         return this;
47908     },
47909     
47910
47911     
47912     
47913     
47914      /**
47915      * Find any element that has been added to a form, using it's ID or name
47916      * This can include framesets, columns etc. along with regular fields..
47917      * @param {String} id - id or name to find.
47918      
47919      * @return {Element} e - or false if nothing found.
47920      */
47921     findbyId : function(id)
47922     {
47923         var ret = false;
47924         if (!id) {
47925             return ret;
47926         }
47927         Roo.each(this.allItems, function(f){
47928             if (f.id == id || f.name == id ){
47929                 ret = f;
47930                 return false;
47931             }
47932         });
47933         return ret;
47934     },
47935
47936     
47937     
47938     /**
47939      * Render this form into the passed container. This should only be called once!
47940      * @param {String/HTMLElement/Element} container The element this component should be rendered into
47941      * @return {Form} this
47942      */
47943     render : function(ct)
47944     {
47945         
47946         
47947         
47948         ct = Roo.get(ct);
47949         var o = this.autoCreate || {
47950             tag: 'form',
47951             method : this.method || 'POST',
47952             id : this.id || Roo.id()
47953         };
47954         this.initEl(ct.createChild(o));
47955
47956         this.root.render(this.el);
47957         
47958        
47959              
47960         this.items.each(function(f){
47961             f.render('x-form-el-'+f.id);
47962         });
47963
47964         if(this.buttons.length > 0){
47965             // tables are required to maintain order and for correct IE layout
47966             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
47967                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
47968                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
47969             }}, null, true);
47970             var tr = tb.getElementsByTagName('tr')[0];
47971             for(var i = 0, len = this.buttons.length; i < len; i++) {
47972                 var b = this.buttons[i];
47973                 var td = document.createElement('td');
47974                 td.className = 'x-form-btn-td';
47975                 b.render(tr.appendChild(td));
47976             }
47977         }
47978         if(this.monitorValid){ // initialize after render
47979             this.startMonitoring();
47980         }
47981         this.fireEvent('rendered', this);
47982         return this;
47983     },
47984
47985     /**
47986      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
47987      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
47988      * object or a valid Roo.DomHelper element config
47989      * @param {Function} handler The function called when the button is clicked
47990      * @param {Object} scope (optional) The scope of the handler function
47991      * @return {Roo.Button}
47992      */
47993     addButton : function(config, handler, scope){
47994         var bc = {
47995             handler: handler,
47996             scope: scope,
47997             minWidth: this.minButtonWidth,
47998             hideParent:true
47999         };
48000         if(typeof config == "string"){
48001             bc.text = config;
48002         }else{
48003             Roo.apply(bc, config);
48004         }
48005         var btn = new Roo.Button(null, bc);
48006         this.buttons.push(btn);
48007         return btn;
48008     },
48009
48010      /**
48011      * Adds a series of form elements (using the xtype property as the factory method.
48012      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48013      * @param {Object} config 
48014      */
48015     
48016     addxtype : function()
48017     {
48018         var ar = Array.prototype.slice.call(arguments, 0);
48019         var ret = false;
48020         for(var i = 0; i < ar.length; i++) {
48021             if (!ar[i]) {
48022                 continue; // skip -- if this happends something invalid got sent, we 
48023                 // should ignore it, as basically that interface element will not show up
48024                 // and that should be pretty obvious!!
48025             }
48026             
48027             if (Roo.form[ar[i].xtype]) {
48028                 ar[i].form = this;
48029                 var fe = Roo.factory(ar[i], Roo.form);
48030                 if (!ret) {
48031                     ret = fe;
48032                 }
48033                 fe.form = this;
48034                 if (fe.store) {
48035                     fe.store.form = this;
48036                 }
48037                 if (fe.isLayout) {  
48038                          
48039                     this.start(fe);
48040                     this.allItems.push(fe);
48041                     if (fe.items && fe.addxtype) {
48042                         fe.addxtype.apply(fe, fe.items);
48043                         delete fe.items;
48044                     }
48045                      this.end();
48046                     continue;
48047                 }
48048                 
48049                 
48050                  
48051                 this.add(fe);
48052               //  console.log('adding ' + ar[i].xtype);
48053             }
48054             if (ar[i].xtype == 'Button') {  
48055                 //console.log('adding button');
48056                 //console.log(ar[i]);
48057                 this.addButton(ar[i]);
48058                 this.allItems.push(fe);
48059                 continue;
48060             }
48061             
48062             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48063                 alert('end is not supported on xtype any more, use items');
48064             //    this.end();
48065             //    //console.log('adding end');
48066             }
48067             
48068         }
48069         return ret;
48070     },
48071     
48072     /**
48073      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48074      * option "monitorValid"
48075      */
48076     startMonitoring : function(){
48077         if(!this.bound){
48078             this.bound = true;
48079             Roo.TaskMgr.start({
48080                 run : this.bindHandler,
48081                 interval : this.monitorPoll || 200,
48082                 scope: this
48083             });
48084         }
48085     },
48086
48087     /**
48088      * Stops monitoring of the valid state of this form
48089      */
48090     stopMonitoring : function(){
48091         this.bound = false;
48092     },
48093
48094     // private
48095     bindHandler : function(){
48096         if(!this.bound){
48097             return false; // stops binding
48098         }
48099         var valid = true;
48100         this.items.each(function(f){
48101             if(!f.isValid(true)){
48102                 valid = false;
48103                 return false;
48104             }
48105         });
48106         for(var i = 0, len = this.buttons.length; i < len; i++){
48107             var btn = this.buttons[i];
48108             if(btn.formBind === true && btn.disabled === valid){
48109                 btn.setDisabled(!valid);
48110             }
48111         }
48112         this.fireEvent('clientvalidation', this, valid);
48113     }
48114     
48115     
48116     
48117     
48118     
48119     
48120     
48121     
48122 });
48123
48124
48125 // back compat
48126 Roo.Form = Roo.form.Form;
48127 /*
48128  * Based on:
48129  * Ext JS Library 1.1.1
48130  * Copyright(c) 2006-2007, Ext JS, LLC.
48131  *
48132  * Originally Released Under LGPL - original licence link has changed is not relivant.
48133  *
48134  * Fork - LGPL
48135  * <script type="text/javascript">
48136  */
48137
48138 // as we use this in bootstrap.
48139 Roo.namespace('Roo.form');
48140  /**
48141  * @class Roo.form.Action
48142  * Internal Class used to handle form actions
48143  * @constructor
48144  * @param {Roo.form.BasicForm} el The form element or its id
48145  * @param {Object} config Configuration options
48146  */
48147
48148  
48149  
48150 // define the action interface
48151 Roo.form.Action = function(form, options){
48152     this.form = form;
48153     this.options = options || {};
48154 };
48155 /**
48156  * Client Validation Failed
48157  * @const 
48158  */
48159 Roo.form.Action.CLIENT_INVALID = 'client';
48160 /**
48161  * Server Validation Failed
48162  * @const 
48163  */
48164 Roo.form.Action.SERVER_INVALID = 'server';
48165  /**
48166  * Connect to Server Failed
48167  * @const 
48168  */
48169 Roo.form.Action.CONNECT_FAILURE = 'connect';
48170 /**
48171  * Reading Data from Server Failed
48172  * @const 
48173  */
48174 Roo.form.Action.LOAD_FAILURE = 'load';
48175
48176 Roo.form.Action.prototype = {
48177     type : 'default',
48178     failureType : undefined,
48179     response : undefined,
48180     result : undefined,
48181
48182     // interface method
48183     run : function(options){
48184
48185     },
48186
48187     // interface method
48188     success : function(response){
48189
48190     },
48191
48192     // interface method
48193     handleResponse : function(response){
48194
48195     },
48196
48197     // default connection failure
48198     failure : function(response){
48199         
48200         this.response = response;
48201         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48202         this.form.afterAction(this, false);
48203     },
48204
48205     processResponse : function(response){
48206         this.response = response;
48207         if(!response.responseText){
48208             return true;
48209         }
48210         this.result = this.handleResponse(response);
48211         return this.result;
48212     },
48213
48214     // utility functions used internally
48215     getUrl : function(appendParams){
48216         var url = this.options.url || this.form.url || this.form.el.dom.action;
48217         if(appendParams){
48218             var p = this.getParams();
48219             if(p){
48220                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48221             }
48222         }
48223         return url;
48224     },
48225
48226     getMethod : function(){
48227         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48228     },
48229
48230     getParams : function(){
48231         var bp = this.form.baseParams;
48232         var p = this.options.params;
48233         if(p){
48234             if(typeof p == "object"){
48235                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48236             }else if(typeof p == 'string' && bp){
48237                 p += '&' + Roo.urlEncode(bp);
48238             }
48239         }else if(bp){
48240             p = Roo.urlEncode(bp);
48241         }
48242         return p;
48243     },
48244
48245     createCallback : function(){
48246         return {
48247             success: this.success,
48248             failure: this.failure,
48249             scope: this,
48250             timeout: (this.form.timeout*1000),
48251             upload: this.form.fileUpload ? this.success : undefined
48252         };
48253     }
48254 };
48255
48256 Roo.form.Action.Submit = function(form, options){
48257     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48258 };
48259
48260 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48261     type : 'submit',
48262
48263     haveProgress : false,
48264     uploadComplete : false,
48265     
48266     // uploadProgress indicator.
48267     uploadProgress : function()
48268     {
48269         if (!this.form.progressUrl) {
48270             return;
48271         }
48272         
48273         if (!this.haveProgress) {
48274             Roo.MessageBox.progress("Uploading", "Uploading");
48275         }
48276         if (this.uploadComplete) {
48277            Roo.MessageBox.hide();
48278            return;
48279         }
48280         
48281         this.haveProgress = true;
48282    
48283         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48284         
48285         var c = new Roo.data.Connection();
48286         c.request({
48287             url : this.form.progressUrl,
48288             params: {
48289                 id : uid
48290             },
48291             method: 'GET',
48292             success : function(req){
48293                //console.log(data);
48294                 var rdata = false;
48295                 var edata;
48296                 try  {
48297                    rdata = Roo.decode(req.responseText)
48298                 } catch (e) {
48299                     Roo.log("Invalid data from server..");
48300                     Roo.log(edata);
48301                     return;
48302                 }
48303                 if (!rdata || !rdata.success) {
48304                     Roo.log(rdata);
48305                     Roo.MessageBox.alert(Roo.encode(rdata));
48306                     return;
48307                 }
48308                 var data = rdata.data;
48309                 
48310                 if (this.uploadComplete) {
48311                    Roo.MessageBox.hide();
48312                    return;
48313                 }
48314                    
48315                 if (data){
48316                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48317                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48318                     );
48319                 }
48320                 this.uploadProgress.defer(2000,this);
48321             },
48322        
48323             failure: function(data) {
48324                 Roo.log('progress url failed ');
48325                 Roo.log(data);
48326             },
48327             scope : this
48328         });
48329            
48330     },
48331     
48332     
48333     run : function()
48334     {
48335         // run get Values on the form, so it syncs any secondary forms.
48336         this.form.getValues();
48337         
48338         var o = this.options;
48339         var method = this.getMethod();
48340         var isPost = method == 'POST';
48341         if(o.clientValidation === false || this.form.isValid()){
48342             
48343             if (this.form.progressUrl) {
48344                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48345                     (new Date() * 1) + '' + Math.random());
48346                     
48347             } 
48348             
48349             
48350             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48351                 form:this.form.el.dom,
48352                 url:this.getUrl(!isPost),
48353                 method: method,
48354                 params:isPost ? this.getParams() : null,
48355                 isUpload: this.form.fileUpload,
48356                 formData : this.form.formData
48357             }));
48358             
48359             this.uploadProgress();
48360
48361         }else if (o.clientValidation !== false){ // client validation failed
48362             this.failureType = Roo.form.Action.CLIENT_INVALID;
48363             this.form.afterAction(this, false);
48364         }
48365     },
48366
48367     success : function(response)
48368     {
48369         this.uploadComplete= true;
48370         if (this.haveProgress) {
48371             Roo.MessageBox.hide();
48372         }
48373         
48374         
48375         var result = this.processResponse(response);
48376         if(result === true || result.success){
48377             this.form.afterAction(this, true);
48378             return;
48379         }
48380         if(result.errors){
48381             this.form.markInvalid(result.errors);
48382             this.failureType = Roo.form.Action.SERVER_INVALID;
48383         }
48384         this.form.afterAction(this, false);
48385     },
48386     failure : function(response)
48387     {
48388         this.uploadComplete= true;
48389         if (this.haveProgress) {
48390             Roo.MessageBox.hide();
48391         }
48392         
48393         this.response = response;
48394         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48395         this.form.afterAction(this, false);
48396     },
48397     
48398     handleResponse : function(response){
48399         if(this.form.errorReader){
48400             var rs = this.form.errorReader.read(response);
48401             var errors = [];
48402             if(rs.records){
48403                 for(var i = 0, len = rs.records.length; i < len; i++) {
48404                     var r = rs.records[i];
48405                     errors[i] = r.data;
48406                 }
48407             }
48408             if(errors.length < 1){
48409                 errors = null;
48410             }
48411             return {
48412                 success : rs.success,
48413                 errors : errors
48414             };
48415         }
48416         var ret = false;
48417         try {
48418             ret = Roo.decode(response.responseText);
48419         } catch (e) {
48420             ret = {
48421                 success: false,
48422                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48423                 errors : []
48424             };
48425         }
48426         return ret;
48427         
48428     }
48429 });
48430
48431
48432 Roo.form.Action.Load = function(form, options){
48433     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48434     this.reader = this.form.reader;
48435 };
48436
48437 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48438     type : 'load',
48439
48440     run : function(){
48441         
48442         Roo.Ajax.request(Roo.apply(
48443                 this.createCallback(), {
48444                     method:this.getMethod(),
48445                     url:this.getUrl(false),
48446                     params:this.getParams()
48447         }));
48448     },
48449
48450     success : function(response){
48451         
48452         var result = this.processResponse(response);
48453         if(result === true || !result.success || !result.data){
48454             this.failureType = Roo.form.Action.LOAD_FAILURE;
48455             this.form.afterAction(this, false);
48456             return;
48457         }
48458         this.form.clearInvalid();
48459         this.form.setValues(result.data);
48460         this.form.afterAction(this, true);
48461     },
48462
48463     handleResponse : function(response){
48464         if(this.form.reader){
48465             var rs = this.form.reader.read(response);
48466             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48467             return {
48468                 success : rs.success,
48469                 data : data
48470             };
48471         }
48472         return Roo.decode(response.responseText);
48473     }
48474 });
48475
48476 Roo.form.Action.ACTION_TYPES = {
48477     'load' : Roo.form.Action.Load,
48478     'submit' : Roo.form.Action.Submit
48479 };/*
48480  * Based on:
48481  * Ext JS Library 1.1.1
48482  * Copyright(c) 2006-2007, Ext JS, LLC.
48483  *
48484  * Originally Released Under LGPL - original licence link has changed is not relivant.
48485  *
48486  * Fork - LGPL
48487  * <script type="text/javascript">
48488  */
48489  
48490 /**
48491  * @class Roo.form.Layout
48492  * @extends Roo.Component
48493  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48494  * @constructor
48495  * @param {Object} config Configuration options
48496  */
48497 Roo.form.Layout = function(config){
48498     var xitems = [];
48499     if (config.items) {
48500         xitems = config.items;
48501         delete config.items;
48502     }
48503     Roo.form.Layout.superclass.constructor.call(this, config);
48504     this.stack = [];
48505     Roo.each(xitems, this.addxtype, this);
48506      
48507 };
48508
48509 Roo.extend(Roo.form.Layout, Roo.Component, {
48510     /**
48511      * @cfg {String/Object} autoCreate
48512      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48513      */
48514     /**
48515      * @cfg {String/Object/Function} style
48516      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48517      * a function which returns such a specification.
48518      */
48519     /**
48520      * @cfg {String} labelAlign
48521      * Valid values are "left," "top" and "right" (defaults to "left")
48522      */
48523     /**
48524      * @cfg {Number} labelWidth
48525      * Fixed width in pixels of all field labels (defaults to undefined)
48526      */
48527     /**
48528      * @cfg {Boolean} clear
48529      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48530      */
48531     clear : true,
48532     /**
48533      * @cfg {String} labelSeparator
48534      * The separator to use after field labels (defaults to ':')
48535      */
48536     labelSeparator : ':',
48537     /**
48538      * @cfg {Boolean} hideLabels
48539      * True to suppress the display of field labels in this layout (defaults to false)
48540      */
48541     hideLabels : false,
48542
48543     // private
48544     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48545     
48546     isLayout : true,
48547     
48548     // private
48549     onRender : function(ct, position){
48550         if(this.el){ // from markup
48551             this.el = Roo.get(this.el);
48552         }else {  // generate
48553             var cfg = this.getAutoCreate();
48554             this.el = ct.createChild(cfg, position);
48555         }
48556         if(this.style){
48557             this.el.applyStyles(this.style);
48558         }
48559         if(this.labelAlign){
48560             this.el.addClass('x-form-label-'+this.labelAlign);
48561         }
48562         if(this.hideLabels){
48563             this.labelStyle = "display:none";
48564             this.elementStyle = "padding-left:0;";
48565         }else{
48566             if(typeof this.labelWidth == 'number'){
48567                 this.labelStyle = "width:"+this.labelWidth+"px;";
48568                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48569             }
48570             if(this.labelAlign == 'top'){
48571                 this.labelStyle = "width:auto;";
48572                 this.elementStyle = "padding-left:0;";
48573             }
48574         }
48575         var stack = this.stack;
48576         var slen = stack.length;
48577         if(slen > 0){
48578             if(!this.fieldTpl){
48579                 var t = new Roo.Template(
48580                     '<div class="x-form-item {5}">',
48581                         '<label for="{0}" style="{2}">{1}{4}</label>',
48582                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48583                         '</div>',
48584                     '</div><div class="x-form-clear-left"></div>'
48585                 );
48586                 t.disableFormats = true;
48587                 t.compile();
48588                 Roo.form.Layout.prototype.fieldTpl = t;
48589             }
48590             for(var i = 0; i < slen; i++) {
48591                 if(stack[i].isFormField){
48592                     this.renderField(stack[i]);
48593                 }else{
48594                     this.renderComponent(stack[i]);
48595                 }
48596             }
48597         }
48598         if(this.clear){
48599             this.el.createChild({cls:'x-form-clear'});
48600         }
48601     },
48602
48603     // private
48604     renderField : function(f){
48605         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
48606                f.id, //0
48607                f.fieldLabel, //1
48608                f.labelStyle||this.labelStyle||'', //2
48609                this.elementStyle||'', //3
48610                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
48611                f.itemCls||this.itemCls||''  //5
48612        ], true).getPrevSibling());
48613     },
48614
48615     // private
48616     renderComponent : function(c){
48617         c.render(c.isLayout ? this.el : this.el.createChild());    
48618     },
48619     /**
48620      * Adds a object form elements (using the xtype property as the factory method.)
48621      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
48622      * @param {Object} config 
48623      */
48624     addxtype : function(o)
48625     {
48626         // create the lement.
48627         o.form = this.form;
48628         var fe = Roo.factory(o, Roo.form);
48629         this.form.allItems.push(fe);
48630         this.stack.push(fe);
48631         
48632         if (fe.isFormField) {
48633             this.form.items.add(fe);
48634         }
48635          
48636         return fe;
48637     }
48638 });
48639
48640 /**
48641  * @class Roo.form.Column
48642  * @extends Roo.form.Layout
48643  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
48644  * @constructor
48645  * @param {Object} config Configuration options
48646  */
48647 Roo.form.Column = function(config){
48648     Roo.form.Column.superclass.constructor.call(this, config);
48649 };
48650
48651 Roo.extend(Roo.form.Column, Roo.form.Layout, {
48652     /**
48653      * @cfg {Number/String} width
48654      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48655      */
48656     /**
48657      * @cfg {String/Object} autoCreate
48658      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
48659      */
48660
48661     // private
48662     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
48663
48664     // private
48665     onRender : function(ct, position){
48666         Roo.form.Column.superclass.onRender.call(this, ct, position);
48667         if(this.width){
48668             this.el.setWidth(this.width);
48669         }
48670     }
48671 });
48672
48673
48674 /**
48675  * @class Roo.form.Row
48676  * @extends Roo.form.Layout
48677  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
48678  * @constructor
48679  * @param {Object} config Configuration options
48680  */
48681
48682  
48683 Roo.form.Row = function(config){
48684     Roo.form.Row.superclass.constructor.call(this, config);
48685 };
48686  
48687 Roo.extend(Roo.form.Row, Roo.form.Layout, {
48688       /**
48689      * @cfg {Number/String} width
48690      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48691      */
48692     /**
48693      * @cfg {Number/String} height
48694      * The fixed height of the column in pixels or CSS value (defaults to "auto")
48695      */
48696     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
48697     
48698     padWidth : 20,
48699     // private
48700     onRender : function(ct, position){
48701         //console.log('row render');
48702         if(!this.rowTpl){
48703             var t = new Roo.Template(
48704                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
48705                     '<label for="{0}" style="{2}">{1}{4}</label>',
48706                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48707                     '</div>',
48708                 '</div>'
48709             );
48710             t.disableFormats = true;
48711             t.compile();
48712             Roo.form.Layout.prototype.rowTpl = t;
48713         }
48714         this.fieldTpl = this.rowTpl;
48715         
48716         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
48717         var labelWidth = 100;
48718         
48719         if ((this.labelAlign != 'top')) {
48720             if (typeof this.labelWidth == 'number') {
48721                 labelWidth = this.labelWidth
48722             }
48723             this.padWidth =  20 + labelWidth;
48724             
48725         }
48726         
48727         Roo.form.Column.superclass.onRender.call(this, ct, position);
48728         if(this.width){
48729             this.el.setWidth(this.width);
48730         }
48731         if(this.height){
48732             this.el.setHeight(this.height);
48733         }
48734     },
48735     
48736     // private
48737     renderField : function(f){
48738         f.fieldEl = this.fieldTpl.append(this.el, [
48739                f.id, f.fieldLabel,
48740                f.labelStyle||this.labelStyle||'',
48741                this.elementStyle||'',
48742                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
48743                f.itemCls||this.itemCls||'',
48744                f.width ? f.width + this.padWidth : 160 + this.padWidth
48745        ],true);
48746     }
48747 });
48748  
48749
48750 /**
48751  * @class Roo.form.FieldSet
48752  * @extends Roo.form.Layout
48753  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
48754  * @constructor
48755  * @param {Object} config Configuration options
48756  */
48757 Roo.form.FieldSet = function(config){
48758     Roo.form.FieldSet.superclass.constructor.call(this, config);
48759 };
48760
48761 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
48762     /**
48763      * @cfg {String} legend
48764      * The text to display as the legend for the FieldSet (defaults to '')
48765      */
48766     /**
48767      * @cfg {String/Object} autoCreate
48768      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
48769      */
48770
48771     // private
48772     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
48773
48774     // private
48775     onRender : function(ct, position){
48776         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
48777         if(this.legend){
48778             this.setLegend(this.legend);
48779         }
48780     },
48781
48782     // private
48783     setLegend : function(text){
48784         if(this.rendered){
48785             this.el.child('legend').update(text);
48786         }
48787     }
48788 });/*
48789  * Based on:
48790  * Ext JS Library 1.1.1
48791  * Copyright(c) 2006-2007, Ext JS, LLC.
48792  *
48793  * Originally Released Under LGPL - original licence link has changed is not relivant.
48794  *
48795  * Fork - LGPL
48796  * <script type="text/javascript">
48797  */
48798 /**
48799  * @class Roo.form.VTypes
48800  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
48801  * @singleton
48802  */
48803 Roo.form.VTypes = function(){
48804     // closure these in so they are only created once.
48805     var alpha = /^[a-zA-Z_]+$/;
48806     var alphanum = /^[a-zA-Z0-9_]+$/;
48807     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
48808     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
48809
48810     // All these messages and functions are configurable
48811     return {
48812         /**
48813          * The function used to validate email addresses
48814          * @param {String} value The email address
48815          */
48816         'email' : function(v){
48817             return email.test(v);
48818         },
48819         /**
48820          * The error text to display when the email validation function returns false
48821          * @type String
48822          */
48823         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
48824         /**
48825          * The keystroke filter mask to be applied on email input
48826          * @type RegExp
48827          */
48828         'emailMask' : /[a-z0-9_\.\-@]/i,
48829
48830         /**
48831          * The function used to validate URLs
48832          * @param {String} value The URL
48833          */
48834         'url' : function(v){
48835             return url.test(v);
48836         },
48837         /**
48838          * The error text to display when the url validation function returns false
48839          * @type String
48840          */
48841         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
48842         
48843         /**
48844          * The function used to validate alpha values
48845          * @param {String} value The value
48846          */
48847         'alpha' : function(v){
48848             return alpha.test(v);
48849         },
48850         /**
48851          * The error text to display when the alpha validation function returns false
48852          * @type String
48853          */
48854         'alphaText' : 'This field should only contain letters and _',
48855         /**
48856          * The keystroke filter mask to be applied on alpha input
48857          * @type RegExp
48858          */
48859         'alphaMask' : /[a-z_]/i,
48860
48861         /**
48862          * The function used to validate alphanumeric values
48863          * @param {String} value The value
48864          */
48865         'alphanum' : function(v){
48866             return alphanum.test(v);
48867         },
48868         /**
48869          * The error text to display when the alphanumeric validation function returns false
48870          * @type String
48871          */
48872         'alphanumText' : 'This field should only contain letters, numbers and _',
48873         /**
48874          * The keystroke filter mask to be applied on alphanumeric input
48875          * @type RegExp
48876          */
48877         'alphanumMask' : /[a-z0-9_]/i
48878     };
48879 }();//<script type="text/javascript">
48880
48881 /**
48882  * @class Roo.form.FCKeditor
48883  * @extends Roo.form.TextArea
48884  * Wrapper around the FCKEditor http://www.fckeditor.net
48885  * @constructor
48886  * Creates a new FCKeditor
48887  * @param {Object} config Configuration options
48888  */
48889 Roo.form.FCKeditor = function(config){
48890     Roo.form.FCKeditor.superclass.constructor.call(this, config);
48891     this.addEvents({
48892          /**
48893          * @event editorinit
48894          * Fired when the editor is initialized - you can add extra handlers here..
48895          * @param {FCKeditor} this
48896          * @param {Object} the FCK object.
48897          */
48898         editorinit : true
48899     });
48900     
48901     
48902 };
48903 Roo.form.FCKeditor.editors = { };
48904 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
48905 {
48906     //defaultAutoCreate : {
48907     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
48908     //},
48909     // private
48910     /**
48911      * @cfg {Object} fck options - see fck manual for details.
48912      */
48913     fckconfig : false,
48914     
48915     /**
48916      * @cfg {Object} fck toolbar set (Basic or Default)
48917      */
48918     toolbarSet : 'Basic',
48919     /**
48920      * @cfg {Object} fck BasePath
48921      */ 
48922     basePath : '/fckeditor/',
48923     
48924     
48925     frame : false,
48926     
48927     value : '',
48928     
48929    
48930     onRender : function(ct, position)
48931     {
48932         if(!this.el){
48933             this.defaultAutoCreate = {
48934                 tag: "textarea",
48935                 style:"width:300px;height:60px;",
48936                 autocomplete: "new-password"
48937             };
48938         }
48939         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
48940         /*
48941         if(this.grow){
48942             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
48943             if(this.preventScrollbars){
48944                 this.el.setStyle("overflow", "hidden");
48945             }
48946             this.el.setHeight(this.growMin);
48947         }
48948         */
48949         //console.log('onrender' + this.getId() );
48950         Roo.form.FCKeditor.editors[this.getId()] = this;
48951          
48952
48953         this.replaceTextarea() ;
48954         
48955     },
48956     
48957     getEditor : function() {
48958         return this.fckEditor;
48959     },
48960     /**
48961      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
48962      * @param {Mixed} value The value to set
48963      */
48964     
48965     
48966     setValue : function(value)
48967     {
48968         //console.log('setValue: ' + value);
48969         
48970         if(typeof(value) == 'undefined') { // not sure why this is happending...
48971             return;
48972         }
48973         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
48974         
48975         //if(!this.el || !this.getEditor()) {
48976         //    this.value = value;
48977             //this.setValue.defer(100,this,[value]);    
48978         //    return;
48979         //} 
48980         
48981         if(!this.getEditor()) {
48982             return;
48983         }
48984         
48985         this.getEditor().SetData(value);
48986         
48987         //
48988
48989     },
48990
48991     /**
48992      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
48993      * @return {Mixed} value The field value
48994      */
48995     getValue : function()
48996     {
48997         
48998         if (this.frame && this.frame.dom.style.display == 'none') {
48999             return Roo.form.FCKeditor.superclass.getValue.call(this);
49000         }
49001         
49002         if(!this.el || !this.getEditor()) {
49003            
49004            // this.getValue.defer(100,this); 
49005             return this.value;
49006         }
49007        
49008         
49009         var value=this.getEditor().GetData();
49010         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49011         return Roo.form.FCKeditor.superclass.getValue.call(this);
49012         
49013
49014     },
49015
49016     /**
49017      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49018      * @return {Mixed} value The field value
49019      */
49020     getRawValue : function()
49021     {
49022         if (this.frame && this.frame.dom.style.display == 'none') {
49023             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49024         }
49025         
49026         if(!this.el || !this.getEditor()) {
49027             //this.getRawValue.defer(100,this); 
49028             return this.value;
49029             return;
49030         }
49031         
49032         
49033         
49034         var value=this.getEditor().GetData();
49035         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49036         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49037          
49038     },
49039     
49040     setSize : function(w,h) {
49041         
49042         
49043         
49044         //if (this.frame && this.frame.dom.style.display == 'none') {
49045         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49046         //    return;
49047         //}
49048         //if(!this.el || !this.getEditor()) {
49049         //    this.setSize.defer(100,this, [w,h]); 
49050         //    return;
49051         //}
49052         
49053         
49054         
49055         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49056         
49057         this.frame.dom.setAttribute('width', w);
49058         this.frame.dom.setAttribute('height', h);
49059         this.frame.setSize(w,h);
49060         
49061     },
49062     
49063     toggleSourceEdit : function(value) {
49064         
49065       
49066          
49067         this.el.dom.style.display = value ? '' : 'none';
49068         this.frame.dom.style.display = value ?  'none' : '';
49069         
49070     },
49071     
49072     
49073     focus: function(tag)
49074     {
49075         if (this.frame.dom.style.display == 'none') {
49076             return Roo.form.FCKeditor.superclass.focus.call(this);
49077         }
49078         if(!this.el || !this.getEditor()) {
49079             this.focus.defer(100,this, [tag]); 
49080             return;
49081         }
49082         
49083         
49084         
49085         
49086         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49087         this.getEditor().Focus();
49088         if (tgs.length) {
49089             if (!this.getEditor().Selection.GetSelection()) {
49090                 this.focus.defer(100,this, [tag]); 
49091                 return;
49092             }
49093             
49094             
49095             var r = this.getEditor().EditorDocument.createRange();
49096             r.setStart(tgs[0],0);
49097             r.setEnd(tgs[0],0);
49098             this.getEditor().Selection.GetSelection().removeAllRanges();
49099             this.getEditor().Selection.GetSelection().addRange(r);
49100             this.getEditor().Focus();
49101         }
49102         
49103     },
49104     
49105     
49106     
49107     replaceTextarea : function()
49108     {
49109         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49110             return ;
49111         }
49112         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49113         //{
49114             // We must check the elements firstly using the Id and then the name.
49115         var oTextarea = document.getElementById( this.getId() );
49116         
49117         var colElementsByName = document.getElementsByName( this.getId() ) ;
49118          
49119         oTextarea.style.display = 'none' ;
49120
49121         if ( oTextarea.tabIndex ) {            
49122             this.TabIndex = oTextarea.tabIndex ;
49123         }
49124         
49125         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49126         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49127         this.frame = Roo.get(this.getId() + '___Frame')
49128     },
49129     
49130     _getConfigHtml : function()
49131     {
49132         var sConfig = '' ;
49133
49134         for ( var o in this.fckconfig ) {
49135             sConfig += sConfig.length > 0  ? '&amp;' : '';
49136             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49137         }
49138
49139         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49140     },
49141     
49142     
49143     _getIFrameHtml : function()
49144     {
49145         var sFile = 'fckeditor.html' ;
49146         /* no idea what this is about..
49147         try
49148         {
49149             if ( (/fcksource=true/i).test( window.top.location.search ) )
49150                 sFile = 'fckeditor.original.html' ;
49151         }
49152         catch (e) { 
49153         */
49154
49155         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49156         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49157         
49158         
49159         var html = '<iframe id="' + this.getId() +
49160             '___Frame" src="' + sLink +
49161             '" width="' + this.width +
49162             '" height="' + this.height + '"' +
49163             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49164             ' frameborder="0" scrolling="no"></iframe>' ;
49165
49166         return html ;
49167     },
49168     
49169     _insertHtmlBefore : function( html, element )
49170     {
49171         if ( element.insertAdjacentHTML )       {
49172             // IE
49173             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49174         } else { // Gecko
49175             var oRange = document.createRange() ;
49176             oRange.setStartBefore( element ) ;
49177             var oFragment = oRange.createContextualFragment( html );
49178             element.parentNode.insertBefore( oFragment, element ) ;
49179         }
49180     }
49181     
49182     
49183   
49184     
49185     
49186     
49187     
49188
49189 });
49190
49191 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49192
49193 function FCKeditor_OnComplete(editorInstance){
49194     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49195     f.fckEditor = editorInstance;
49196     //console.log("loaded");
49197     f.fireEvent('editorinit', f, editorInstance);
49198
49199   
49200
49201  
49202
49203
49204
49205
49206
49207
49208
49209
49210
49211
49212
49213
49214
49215
49216
49217 //<script type="text/javascript">
49218 /**
49219  * @class Roo.form.GridField
49220  * @extends Roo.form.Field
49221  * Embed a grid (or editable grid into a form)
49222  * STATUS ALPHA
49223  * 
49224  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49225  * it needs 
49226  * xgrid.store = Roo.data.Store
49227  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49228  * xgrid.store.reader = Roo.data.JsonReader 
49229  * 
49230  * 
49231  * @constructor
49232  * Creates a new GridField
49233  * @param {Object} config Configuration options
49234  */
49235 Roo.form.GridField = function(config){
49236     Roo.form.GridField.superclass.constructor.call(this, config);
49237      
49238 };
49239
49240 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49241     /**
49242      * @cfg {Number} width  - used to restrict width of grid..
49243      */
49244     width : 100,
49245     /**
49246      * @cfg {Number} height - used to restrict height of grid..
49247      */
49248     height : 50,
49249      /**
49250      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49251          * 
49252          *}
49253      */
49254     xgrid : false, 
49255     /**
49256      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49257      * {tag: "input", type: "checkbox", autocomplete: "off"})
49258      */
49259    // defaultAutoCreate : { tag: 'div' },
49260     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49261     /**
49262      * @cfg {String} addTitle Text to include for adding a title.
49263      */
49264     addTitle : false,
49265     //
49266     onResize : function(){
49267         Roo.form.Field.superclass.onResize.apply(this, arguments);
49268     },
49269
49270     initEvents : function(){
49271         // Roo.form.Checkbox.superclass.initEvents.call(this);
49272         // has no events...
49273        
49274     },
49275
49276
49277     getResizeEl : function(){
49278         return this.wrap;
49279     },
49280
49281     getPositionEl : function(){
49282         return this.wrap;
49283     },
49284
49285     // private
49286     onRender : function(ct, position){
49287         
49288         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49289         var style = this.style;
49290         delete this.style;
49291         
49292         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49293         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49294         this.viewEl = this.wrap.createChild({ tag: 'div' });
49295         if (style) {
49296             this.viewEl.applyStyles(style);
49297         }
49298         if (this.width) {
49299             this.viewEl.setWidth(this.width);
49300         }
49301         if (this.height) {
49302             this.viewEl.setHeight(this.height);
49303         }
49304         //if(this.inputValue !== undefined){
49305         //this.setValue(this.value);
49306         
49307         
49308         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49309         
49310         
49311         this.grid.render();
49312         this.grid.getDataSource().on('remove', this.refreshValue, this);
49313         this.grid.getDataSource().on('update', this.refreshValue, this);
49314         this.grid.on('afteredit', this.refreshValue, this);
49315  
49316     },
49317      
49318     
49319     /**
49320      * Sets the value of the item. 
49321      * @param {String} either an object  or a string..
49322      */
49323     setValue : function(v){
49324         //this.value = v;
49325         v = v || []; // empty set..
49326         // this does not seem smart - it really only affects memoryproxy grids..
49327         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49328             var ds = this.grid.getDataSource();
49329             // assumes a json reader..
49330             var data = {}
49331             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49332             ds.loadData( data);
49333         }
49334         // clear selection so it does not get stale.
49335         if (this.grid.sm) { 
49336             this.grid.sm.clearSelections();
49337         }
49338         
49339         Roo.form.GridField.superclass.setValue.call(this, v);
49340         this.refreshValue();
49341         // should load data in the grid really....
49342     },
49343     
49344     // private
49345     refreshValue: function() {
49346          var val = [];
49347         this.grid.getDataSource().each(function(r) {
49348             val.push(r.data);
49349         });
49350         this.el.dom.value = Roo.encode(val);
49351     }
49352     
49353      
49354     
49355     
49356 });/*
49357  * Based on:
49358  * Ext JS Library 1.1.1
49359  * Copyright(c) 2006-2007, Ext JS, LLC.
49360  *
49361  * Originally Released Under LGPL - original licence link has changed is not relivant.
49362  *
49363  * Fork - LGPL
49364  * <script type="text/javascript">
49365  */
49366 /**
49367  * @class Roo.form.DisplayField
49368  * @extends Roo.form.Field
49369  * A generic Field to display non-editable data.
49370  * @cfg {Boolean} closable (true|false) default false
49371  * @constructor
49372  * Creates a new Display Field item.
49373  * @param {Object} config Configuration options
49374  */
49375 Roo.form.DisplayField = function(config){
49376     Roo.form.DisplayField.superclass.constructor.call(this, config);
49377     
49378     this.addEvents({
49379         /**
49380          * @event close
49381          * Fires after the click the close btn
49382              * @param {Roo.form.DisplayField} this
49383              */
49384         close : true
49385     });
49386 };
49387
49388 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49389     inputType:      'hidden',
49390     allowBlank:     true,
49391     readOnly:         true,
49392     
49393  
49394     /**
49395      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49396      */
49397     focusClass : undefined,
49398     /**
49399      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49400      */
49401     fieldClass: 'x-form-field',
49402     
49403      /**
49404      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49405      */
49406     valueRenderer: undefined,
49407     
49408     width: 100,
49409     /**
49410      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49411      * {tag: "input", type: "checkbox", autocomplete: "off"})
49412      */
49413      
49414  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49415  
49416     closable : false,
49417     
49418     onResize : function(){
49419         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49420         
49421     },
49422
49423     initEvents : function(){
49424         // Roo.form.Checkbox.superclass.initEvents.call(this);
49425         // has no events...
49426         
49427         if(this.closable){
49428             this.closeEl.on('click', this.onClose, this);
49429         }
49430        
49431     },
49432
49433
49434     getResizeEl : function(){
49435         return this.wrap;
49436     },
49437
49438     getPositionEl : function(){
49439         return this.wrap;
49440     },
49441
49442     // private
49443     onRender : function(ct, position){
49444         
49445         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49446         //if(this.inputValue !== undefined){
49447         this.wrap = this.el.wrap();
49448         
49449         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49450         
49451         if(this.closable){
49452             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49453         }
49454         
49455         if (this.bodyStyle) {
49456             this.viewEl.applyStyles(this.bodyStyle);
49457         }
49458         //this.viewEl.setStyle('padding', '2px');
49459         
49460         this.setValue(this.value);
49461         
49462     },
49463 /*
49464     // private
49465     initValue : Roo.emptyFn,
49466
49467   */
49468
49469         // private
49470     onClick : function(){
49471         
49472     },
49473
49474     /**
49475      * Sets the checked state of the checkbox.
49476      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49477      */
49478     setValue : function(v){
49479         this.value = v;
49480         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49481         // this might be called before we have a dom element..
49482         if (!this.viewEl) {
49483             return;
49484         }
49485         this.viewEl.dom.innerHTML = html;
49486         Roo.form.DisplayField.superclass.setValue.call(this, v);
49487
49488     },
49489     
49490     onClose : function(e)
49491     {
49492         e.preventDefault();
49493         
49494         this.fireEvent('close', this);
49495     }
49496 });/*
49497  * 
49498  * Licence- LGPL
49499  * 
49500  */
49501
49502 /**
49503  * @class Roo.form.DayPicker
49504  * @extends Roo.form.Field
49505  * A Day picker show [M] [T] [W] ....
49506  * @constructor
49507  * Creates a new Day Picker
49508  * @param {Object} config Configuration options
49509  */
49510 Roo.form.DayPicker= function(config){
49511     Roo.form.DayPicker.superclass.constructor.call(this, config);
49512      
49513 };
49514
49515 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49516     /**
49517      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49518      */
49519     focusClass : undefined,
49520     /**
49521      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49522      */
49523     fieldClass: "x-form-field",
49524    
49525     /**
49526      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49527      * {tag: "input", type: "checkbox", autocomplete: "off"})
49528      */
49529     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49530     
49531    
49532     actionMode : 'viewEl', 
49533     //
49534     // private
49535  
49536     inputType : 'hidden',
49537     
49538      
49539     inputElement: false, // real input element?
49540     basedOn: false, // ????
49541     
49542     isFormField: true, // not sure where this is needed!!!!
49543
49544     onResize : function(){
49545         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49546         if(!this.boxLabel){
49547             this.el.alignTo(this.wrap, 'c-c');
49548         }
49549     },
49550
49551     initEvents : function(){
49552         Roo.form.Checkbox.superclass.initEvents.call(this);
49553         this.el.on("click", this.onClick,  this);
49554         this.el.on("change", this.onClick,  this);
49555     },
49556
49557
49558     getResizeEl : function(){
49559         return this.wrap;
49560     },
49561
49562     getPositionEl : function(){
49563         return this.wrap;
49564     },
49565
49566     
49567     // private
49568     onRender : function(ct, position){
49569         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49570        
49571         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49572         
49573         var r1 = '<table><tr>';
49574         var r2 = '<tr class="x-form-daypick-icons">';
49575         for (var i=0; i < 7; i++) {
49576             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49577             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49578         }
49579         
49580         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49581         viewEl.select('img').on('click', this.onClick, this);
49582         this.viewEl = viewEl;   
49583         
49584         
49585         // this will not work on Chrome!!!
49586         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49587         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49588         
49589         
49590           
49591
49592     },
49593
49594     // private
49595     initValue : Roo.emptyFn,
49596
49597     /**
49598      * Returns the checked state of the checkbox.
49599      * @return {Boolean} True if checked, else false
49600      */
49601     getValue : function(){
49602         return this.el.dom.value;
49603         
49604     },
49605
49606         // private
49607     onClick : function(e){ 
49608         //this.setChecked(!this.checked);
49609         Roo.get(e.target).toggleClass('x-menu-item-checked');
49610         this.refreshValue();
49611         //if(this.el.dom.checked != this.checked){
49612         //    this.setValue(this.el.dom.checked);
49613        // }
49614     },
49615     
49616     // private
49617     refreshValue : function()
49618     {
49619         var val = '';
49620         this.viewEl.select('img',true).each(function(e,i,n)  {
49621             val += e.is(".x-menu-item-checked") ? String(n) : '';
49622         });
49623         this.setValue(val, true);
49624     },
49625
49626     /**
49627      * Sets the checked state of the checkbox.
49628      * On is always based on a string comparison between inputValue and the param.
49629      * @param {Boolean/String} value - the value to set 
49630      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
49631      */
49632     setValue : function(v,suppressEvent){
49633         if (!this.el.dom) {
49634             return;
49635         }
49636         var old = this.el.dom.value ;
49637         this.el.dom.value = v;
49638         if (suppressEvent) {
49639             return ;
49640         }
49641          
49642         // update display..
49643         this.viewEl.select('img',true).each(function(e,i,n)  {
49644             
49645             var on = e.is(".x-menu-item-checked");
49646             var newv = v.indexOf(String(n)) > -1;
49647             if (on != newv) {
49648                 e.toggleClass('x-menu-item-checked');
49649             }
49650             
49651         });
49652         
49653         
49654         this.fireEvent('change', this, v, old);
49655         
49656         
49657     },
49658    
49659     // handle setting of hidden value by some other method!!?!?
49660     setFromHidden: function()
49661     {
49662         if(!this.el){
49663             return;
49664         }
49665         //console.log("SET FROM HIDDEN");
49666         //alert('setFrom hidden');
49667         this.setValue(this.el.dom.value);
49668     },
49669     
49670     onDestroy : function()
49671     {
49672         if(this.viewEl){
49673             Roo.get(this.viewEl).remove();
49674         }
49675          
49676         Roo.form.DayPicker.superclass.onDestroy.call(this);
49677     }
49678
49679 });/*
49680  * RooJS Library 1.1.1
49681  * Copyright(c) 2008-2011  Alan Knowles
49682  *
49683  * License - LGPL
49684  */
49685  
49686
49687 /**
49688  * @class Roo.form.ComboCheck
49689  * @extends Roo.form.ComboBox
49690  * A combobox for multiple select items.
49691  *
49692  * FIXME - could do with a reset button..
49693  * 
49694  * @constructor
49695  * Create a new ComboCheck
49696  * @param {Object} config Configuration options
49697  */
49698 Roo.form.ComboCheck = function(config){
49699     Roo.form.ComboCheck.superclass.constructor.call(this, config);
49700     // should verify some data...
49701     // like
49702     // hiddenName = required..
49703     // displayField = required
49704     // valudField == required
49705     var req= [ 'hiddenName', 'displayField', 'valueField' ];
49706     var _t = this;
49707     Roo.each(req, function(e) {
49708         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
49709             throw "Roo.form.ComboCheck : missing value for: " + e;
49710         }
49711     });
49712     
49713     
49714 };
49715
49716 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
49717      
49718      
49719     editable : false,
49720      
49721     selectedClass: 'x-menu-item-checked', 
49722     
49723     // private
49724     onRender : function(ct, position){
49725         var _t = this;
49726         
49727         
49728         
49729         if(!this.tpl){
49730             var cls = 'x-combo-list';
49731
49732             
49733             this.tpl =  new Roo.Template({
49734                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
49735                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
49736                    '<span>{' + this.displayField + '}</span>' +
49737                     '</div>' 
49738                 
49739             });
49740         }
49741  
49742         
49743         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
49744         this.view.singleSelect = false;
49745         this.view.multiSelect = true;
49746         this.view.toggleSelect = true;
49747         this.pageTb.add(new Roo.Toolbar.Fill(), {
49748             
49749             text: 'Done',
49750             handler: function()
49751             {
49752                 _t.collapse();
49753             }
49754         });
49755     },
49756     
49757     onViewOver : function(e, t){
49758         // do nothing...
49759         return;
49760         
49761     },
49762     
49763     onViewClick : function(doFocus,index){
49764         return;
49765         
49766     },
49767     select: function () {
49768         //Roo.log("SELECT CALLED");
49769     },
49770      
49771     selectByValue : function(xv, scrollIntoView){
49772         var ar = this.getValueArray();
49773         var sels = [];
49774         
49775         Roo.each(ar, function(v) {
49776             if(v === undefined || v === null){
49777                 return;
49778             }
49779             var r = this.findRecord(this.valueField, v);
49780             if(r){
49781                 sels.push(this.store.indexOf(r))
49782                 
49783             }
49784         },this);
49785         this.view.select(sels);
49786         return false;
49787     },
49788     
49789     
49790     
49791     onSelect : function(record, index){
49792        // Roo.log("onselect Called");
49793        // this is only called by the clear button now..
49794         this.view.clearSelections();
49795         this.setValue('[]');
49796         if (this.value != this.valueBefore) {
49797             this.fireEvent('change', this, this.value, this.valueBefore);
49798             this.valueBefore = this.value;
49799         }
49800     },
49801     getValueArray : function()
49802     {
49803         var ar = [] ;
49804         
49805         try {
49806             //Roo.log(this.value);
49807             if (typeof(this.value) == 'undefined') {
49808                 return [];
49809             }
49810             var ar = Roo.decode(this.value);
49811             return  ar instanceof Array ? ar : []; //?? valid?
49812             
49813         } catch(e) {
49814             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
49815             return [];
49816         }
49817          
49818     },
49819     expand : function ()
49820     {
49821         
49822         Roo.form.ComboCheck.superclass.expand.call(this);
49823         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
49824         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
49825         
49826
49827     },
49828     
49829     collapse : function(){
49830         Roo.form.ComboCheck.superclass.collapse.call(this);
49831         var sl = this.view.getSelectedIndexes();
49832         var st = this.store;
49833         var nv = [];
49834         var tv = [];
49835         var r;
49836         Roo.each(sl, function(i) {
49837             r = st.getAt(i);
49838             nv.push(r.get(this.valueField));
49839         },this);
49840         this.setValue(Roo.encode(nv));
49841         if (this.value != this.valueBefore) {
49842
49843             this.fireEvent('change', this, this.value, this.valueBefore);
49844             this.valueBefore = this.value;
49845         }
49846         
49847     },
49848     
49849     setValue : function(v){
49850         // Roo.log(v);
49851         this.value = v;
49852         
49853         var vals = this.getValueArray();
49854         var tv = [];
49855         Roo.each(vals, function(k) {
49856             var r = this.findRecord(this.valueField, k);
49857             if(r){
49858                 tv.push(r.data[this.displayField]);
49859             }else if(this.valueNotFoundText !== undefined){
49860                 tv.push( this.valueNotFoundText );
49861             }
49862         },this);
49863        // Roo.log(tv);
49864         
49865         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
49866         this.hiddenField.value = v;
49867         this.value = v;
49868     }
49869     
49870 });/*
49871  * Based on:
49872  * Ext JS Library 1.1.1
49873  * Copyright(c) 2006-2007, Ext JS, LLC.
49874  *
49875  * Originally Released Under LGPL - original licence link has changed is not relivant.
49876  *
49877  * Fork - LGPL
49878  * <script type="text/javascript">
49879  */
49880  
49881 /**
49882  * @class Roo.form.Signature
49883  * @extends Roo.form.Field
49884  * Signature field.  
49885  * @constructor
49886  * 
49887  * @param {Object} config Configuration options
49888  */
49889
49890 Roo.form.Signature = function(config){
49891     Roo.form.Signature.superclass.constructor.call(this, config);
49892     
49893     this.addEvents({// not in used??
49894          /**
49895          * @event confirm
49896          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
49897              * @param {Roo.form.Signature} combo This combo box
49898              */
49899         'confirm' : true,
49900         /**
49901          * @event reset
49902          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
49903              * @param {Roo.form.ComboBox} combo This combo box
49904              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
49905              */
49906         'reset' : true
49907     });
49908 };
49909
49910 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
49911     /**
49912      * @cfg {Object} labels Label to use when rendering a form.
49913      * defaults to 
49914      * labels : { 
49915      *      clear : "Clear",
49916      *      confirm : "Confirm"
49917      *  }
49918      */
49919     labels : { 
49920         clear : "Clear",
49921         confirm : "Confirm"
49922     },
49923     /**
49924      * @cfg {Number} width The signature panel width (defaults to 300)
49925      */
49926     width: 300,
49927     /**
49928      * @cfg {Number} height The signature panel height (defaults to 100)
49929      */
49930     height : 100,
49931     /**
49932      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
49933      */
49934     allowBlank : false,
49935     
49936     //private
49937     // {Object} signPanel The signature SVG panel element (defaults to {})
49938     signPanel : {},
49939     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
49940     isMouseDown : false,
49941     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
49942     isConfirmed : false,
49943     // {String} signatureTmp SVG mapping string (defaults to empty string)
49944     signatureTmp : '',
49945     
49946     
49947     defaultAutoCreate : { // modified by initCompnoent..
49948         tag: "input",
49949         type:"hidden"
49950     },
49951
49952     // private
49953     onRender : function(ct, position){
49954         
49955         Roo.form.Signature.superclass.onRender.call(this, ct, position);
49956         
49957         this.wrap = this.el.wrap({
49958             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
49959         });
49960         
49961         this.createToolbar(this);
49962         this.signPanel = this.wrap.createChild({
49963                 tag: 'div',
49964                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
49965             }, this.el
49966         );
49967             
49968         this.svgID = Roo.id();
49969         this.svgEl = this.signPanel.createChild({
49970               xmlns : 'http://www.w3.org/2000/svg',
49971               tag : 'svg',
49972               id : this.svgID + "-svg",
49973               width: this.width,
49974               height: this.height,
49975               viewBox: '0 0 '+this.width+' '+this.height,
49976               cn : [
49977                 {
49978                     tag: "rect",
49979                     id: this.svgID + "-svg-r",
49980                     width: this.width,
49981                     height: this.height,
49982                     fill: "#ffa"
49983                 },
49984                 {
49985                     tag: "line",
49986                     id: this.svgID + "-svg-l",
49987                     x1: "0", // start
49988                     y1: (this.height*0.8), // start set the line in 80% of height
49989                     x2: this.width, // end
49990                     y2: (this.height*0.8), // end set the line in 80% of height
49991                     'stroke': "#666",
49992                     'stroke-width': "1",
49993                     'stroke-dasharray': "3",
49994                     'shape-rendering': "crispEdges",
49995                     'pointer-events': "none"
49996                 },
49997                 {
49998                     tag: "path",
49999                     id: this.svgID + "-svg-p",
50000                     'stroke': "navy",
50001                     'stroke-width': "3",
50002                     'fill': "none",
50003                     'pointer-events': 'none'
50004                 }
50005               ]
50006         });
50007         this.createSVG();
50008         this.svgBox = this.svgEl.dom.getScreenCTM();
50009     },
50010     createSVG : function(){ 
50011         var svg = this.signPanel;
50012         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50013         var t = this;
50014
50015         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50016         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50017         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50018         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50019         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50020         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50021         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50022         
50023     },
50024     isTouchEvent : function(e){
50025         return e.type.match(/^touch/);
50026     },
50027     getCoords : function (e) {
50028         var pt    = this.svgEl.dom.createSVGPoint();
50029         pt.x = e.clientX; 
50030         pt.y = e.clientY;
50031         if (this.isTouchEvent(e)) {
50032             pt.x =  e.targetTouches[0].clientX;
50033             pt.y = e.targetTouches[0].clientY;
50034         }
50035         var a = this.svgEl.dom.getScreenCTM();
50036         var b = a.inverse();
50037         var mx = pt.matrixTransform(b);
50038         return mx.x + ',' + mx.y;
50039     },
50040     //mouse event headler 
50041     down : function (e) {
50042         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50043         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50044         
50045         this.isMouseDown = true;
50046         
50047         e.preventDefault();
50048     },
50049     move : function (e) {
50050         if (this.isMouseDown) {
50051             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50052             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50053         }
50054         
50055         e.preventDefault();
50056     },
50057     up : function (e) {
50058         this.isMouseDown = false;
50059         var sp = this.signatureTmp.split(' ');
50060         
50061         if(sp.length > 1){
50062             if(!sp[sp.length-2].match(/^L/)){
50063                 sp.pop();
50064                 sp.pop();
50065                 sp.push("");
50066                 this.signatureTmp = sp.join(" ");
50067             }
50068         }
50069         if(this.getValue() != this.signatureTmp){
50070             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50071             this.isConfirmed = false;
50072         }
50073         e.preventDefault();
50074     },
50075     
50076     /**
50077      * Protected method that will not generally be called directly. It
50078      * is called when the editor creates its toolbar. Override this method if you need to
50079      * add custom toolbar buttons.
50080      * @param {HtmlEditor} editor
50081      */
50082     createToolbar : function(editor){
50083          function btn(id, toggle, handler){
50084             var xid = fid + '-'+ id ;
50085             return {
50086                 id : xid,
50087                 cmd : id,
50088                 cls : 'x-btn-icon x-edit-'+id,
50089                 enableToggle:toggle !== false,
50090                 scope: editor, // was editor...
50091                 handler:handler||editor.relayBtnCmd,
50092                 clickEvent:'mousedown',
50093                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50094                 tabIndex:-1
50095             };
50096         }
50097         
50098         
50099         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50100         this.tb = tb;
50101         this.tb.add(
50102            {
50103                 cls : ' x-signature-btn x-signature-'+id,
50104                 scope: editor, // was editor...
50105                 handler: this.reset,
50106                 clickEvent:'mousedown',
50107                 text: this.labels.clear
50108             },
50109             {
50110                  xtype : 'Fill',
50111                  xns: Roo.Toolbar
50112             }, 
50113             {
50114                 cls : '  x-signature-btn x-signature-'+id,
50115                 scope: editor, // was editor...
50116                 handler: this.confirmHandler,
50117                 clickEvent:'mousedown',
50118                 text: this.labels.confirm
50119             }
50120         );
50121     
50122     },
50123     //public
50124     /**
50125      * when user is clicked confirm then show this image.....
50126      * 
50127      * @return {String} Image Data URI
50128      */
50129     getImageDataURI : function(){
50130         var svg = this.svgEl.dom.parentNode.innerHTML;
50131         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50132         return src; 
50133     },
50134     /**
50135      * 
50136      * @return {Boolean} this.isConfirmed
50137      */
50138     getConfirmed : function(){
50139         return this.isConfirmed;
50140     },
50141     /**
50142      * 
50143      * @return {Number} this.width
50144      */
50145     getWidth : function(){
50146         return this.width;
50147     },
50148     /**
50149      * 
50150      * @return {Number} this.height
50151      */
50152     getHeight : function(){
50153         return this.height;
50154     },
50155     // private
50156     getSignature : function(){
50157         return this.signatureTmp;
50158     },
50159     // private
50160     reset : function(){
50161         this.signatureTmp = '';
50162         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50163         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50164         this.isConfirmed = false;
50165         Roo.form.Signature.superclass.reset.call(this);
50166     },
50167     setSignature : function(s){
50168         this.signatureTmp = s;
50169         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50170         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50171         this.setValue(s);
50172         this.isConfirmed = false;
50173         Roo.form.Signature.superclass.reset.call(this);
50174     }, 
50175     test : function(){
50176 //        Roo.log(this.signPanel.dom.contentWindow.up())
50177     },
50178     //private
50179     setConfirmed : function(){
50180         
50181         
50182         
50183 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50184     },
50185     // private
50186     confirmHandler : function(){
50187         if(!this.getSignature()){
50188             return;
50189         }
50190         
50191         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50192         this.setValue(this.getSignature());
50193         this.isConfirmed = true;
50194         
50195         this.fireEvent('confirm', this);
50196     },
50197     // private
50198     // Subclasses should provide the validation implementation by overriding this
50199     validateValue : function(value){
50200         if(this.allowBlank){
50201             return true;
50202         }
50203         
50204         if(this.isConfirmed){
50205             return true;
50206         }
50207         return false;
50208     }
50209 });/*
50210  * Based on:
50211  * Ext JS Library 1.1.1
50212  * Copyright(c) 2006-2007, Ext JS, LLC.
50213  *
50214  * Originally Released Under LGPL - original licence link has changed is not relivant.
50215  *
50216  * Fork - LGPL
50217  * <script type="text/javascript">
50218  */
50219  
50220
50221 /**
50222  * @class Roo.form.ComboBox
50223  * @extends Roo.form.TriggerField
50224  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50225  * @constructor
50226  * Create a new ComboBox.
50227  * @param {Object} config Configuration options
50228  */
50229 Roo.form.Select = function(config){
50230     Roo.form.Select.superclass.constructor.call(this, config);
50231      
50232 };
50233
50234 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50235     /**
50236      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50237      */
50238     /**
50239      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50240      * rendering into an Roo.Editor, defaults to false)
50241      */
50242     /**
50243      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50244      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50245      */
50246     /**
50247      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50248      */
50249     /**
50250      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50251      * the dropdown list (defaults to undefined, with no header element)
50252      */
50253
50254      /**
50255      * @cfg {String/Roo.Template} tpl The template to use to render the output
50256      */
50257      
50258     // private
50259     defaultAutoCreate : {tag: "select"  },
50260     /**
50261      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50262      */
50263     listWidth: undefined,
50264     /**
50265      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50266      * mode = 'remote' or 'text' if mode = 'local')
50267      */
50268     displayField: undefined,
50269     /**
50270      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50271      * mode = 'remote' or 'value' if mode = 'local'). 
50272      * Note: use of a valueField requires the user make a selection
50273      * in order for a value to be mapped.
50274      */
50275     valueField: undefined,
50276     
50277     
50278     /**
50279      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50280      * field's data value (defaults to the underlying DOM element's name)
50281      */
50282     hiddenName: undefined,
50283     /**
50284      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50285      */
50286     listClass: '',
50287     /**
50288      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50289      */
50290     selectedClass: 'x-combo-selected',
50291     /**
50292      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50293      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50294      * which displays a downward arrow icon).
50295      */
50296     triggerClass : 'x-form-arrow-trigger',
50297     /**
50298      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50299      */
50300     shadow:'sides',
50301     /**
50302      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50303      * anchor positions (defaults to 'tl-bl')
50304      */
50305     listAlign: 'tl-bl?',
50306     /**
50307      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50308      */
50309     maxHeight: 300,
50310     /**
50311      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50312      * query specified by the allQuery config option (defaults to 'query')
50313      */
50314     triggerAction: 'query',
50315     /**
50316      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50317      * (defaults to 4, does not apply if editable = false)
50318      */
50319     minChars : 4,
50320     /**
50321      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50322      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50323      */
50324     typeAhead: false,
50325     /**
50326      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50327      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50328      */
50329     queryDelay: 500,
50330     /**
50331      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50332      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50333      */
50334     pageSize: 0,
50335     /**
50336      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50337      * when editable = true (defaults to false)
50338      */
50339     selectOnFocus:false,
50340     /**
50341      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50342      */
50343     queryParam: 'query',
50344     /**
50345      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50346      * when mode = 'remote' (defaults to 'Loading...')
50347      */
50348     loadingText: 'Loading...',
50349     /**
50350      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50351      */
50352     resizable: false,
50353     /**
50354      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50355      */
50356     handleHeight : 8,
50357     /**
50358      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50359      * traditional select (defaults to true)
50360      */
50361     editable: true,
50362     /**
50363      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50364      */
50365     allQuery: '',
50366     /**
50367      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50368      */
50369     mode: 'remote',
50370     /**
50371      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50372      * listWidth has a higher value)
50373      */
50374     minListWidth : 70,
50375     /**
50376      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50377      * allow the user to set arbitrary text into the field (defaults to false)
50378      */
50379     forceSelection:false,
50380     /**
50381      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50382      * if typeAhead = true (defaults to 250)
50383      */
50384     typeAheadDelay : 250,
50385     /**
50386      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50387      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50388      */
50389     valueNotFoundText : undefined,
50390     
50391     /**
50392      * @cfg {String} defaultValue The value displayed after loading the store.
50393      */
50394     defaultValue: '',
50395     
50396     /**
50397      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50398      */
50399     blockFocus : false,
50400     
50401     /**
50402      * @cfg {Boolean} disableClear Disable showing of clear button.
50403      */
50404     disableClear : false,
50405     /**
50406      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50407      */
50408     alwaysQuery : false,
50409     
50410     //private
50411     addicon : false,
50412     editicon: false,
50413     
50414     // element that contains real text value.. (when hidden is used..)
50415      
50416     // private
50417     onRender : function(ct, position){
50418         Roo.form.Field.prototype.onRender.call(this, ct, position);
50419         
50420         if(this.store){
50421             this.store.on('beforeload', this.onBeforeLoad, this);
50422             this.store.on('load', this.onLoad, this);
50423             this.store.on('loadexception', this.onLoadException, this);
50424             this.store.load({});
50425         }
50426         
50427         
50428         
50429     },
50430
50431     // private
50432     initEvents : function(){
50433         //Roo.form.ComboBox.superclass.initEvents.call(this);
50434  
50435     },
50436
50437     onDestroy : function(){
50438        
50439         if(this.store){
50440             this.store.un('beforeload', this.onBeforeLoad, this);
50441             this.store.un('load', this.onLoad, this);
50442             this.store.un('loadexception', this.onLoadException, this);
50443         }
50444         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50445     },
50446
50447     // private
50448     fireKey : function(e){
50449         if(e.isNavKeyPress() && !this.list.isVisible()){
50450             this.fireEvent("specialkey", this, e);
50451         }
50452     },
50453
50454     // private
50455     onResize: function(w, h){
50456         
50457         return; 
50458     
50459         
50460     },
50461
50462     /**
50463      * Allow or prevent the user from directly editing the field text.  If false is passed,
50464      * the user will only be able to select from the items defined in the dropdown list.  This method
50465      * is the runtime equivalent of setting the 'editable' config option at config time.
50466      * @param {Boolean} value True to allow the user to directly edit the field text
50467      */
50468     setEditable : function(value){
50469          
50470     },
50471
50472     // private
50473     onBeforeLoad : function(){
50474         
50475         Roo.log("Select before load");
50476         return;
50477     
50478         this.innerList.update(this.loadingText ?
50479                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50480         //this.restrictHeight();
50481         this.selectedIndex = -1;
50482     },
50483
50484     // private
50485     onLoad : function(){
50486
50487     
50488         var dom = this.el.dom;
50489         dom.innerHTML = '';
50490          var od = dom.ownerDocument;
50491          
50492         if (this.emptyText) {
50493             var op = od.createElement('option');
50494             op.setAttribute('value', '');
50495             op.innerHTML = String.format('{0}', this.emptyText);
50496             dom.appendChild(op);
50497         }
50498         if(this.store.getCount() > 0){
50499            
50500             var vf = this.valueField;
50501             var df = this.displayField;
50502             this.store.data.each(function(r) {
50503                 // which colmsn to use... testing - cdoe / title..
50504                 var op = od.createElement('option');
50505                 op.setAttribute('value', r.data[vf]);
50506                 op.innerHTML = String.format('{0}', r.data[df]);
50507                 dom.appendChild(op);
50508             });
50509             if (typeof(this.defaultValue != 'undefined')) {
50510                 this.setValue(this.defaultValue);
50511             }
50512             
50513              
50514         }else{
50515             //this.onEmptyResults();
50516         }
50517         //this.el.focus();
50518     },
50519     // private
50520     onLoadException : function()
50521     {
50522         dom.innerHTML = '';
50523             
50524         Roo.log("Select on load exception");
50525         return;
50526     
50527         this.collapse();
50528         Roo.log(this.store.reader.jsonData);
50529         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50530             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50531         }
50532         
50533         
50534     },
50535     // private
50536     onTypeAhead : function(){
50537          
50538     },
50539
50540     // private
50541     onSelect : function(record, index){
50542         Roo.log('on select?');
50543         return;
50544         if(this.fireEvent('beforeselect', this, record, index) !== false){
50545             this.setFromData(index > -1 ? record.data : false);
50546             this.collapse();
50547             this.fireEvent('select', this, record, index);
50548         }
50549     },
50550
50551     /**
50552      * Returns the currently selected field value or empty string if no value is set.
50553      * @return {String} value The selected value
50554      */
50555     getValue : function(){
50556         var dom = this.el.dom;
50557         this.value = dom.options[dom.selectedIndex].value;
50558         return this.value;
50559         
50560     },
50561
50562     /**
50563      * Clears any text/value currently set in the field
50564      */
50565     clearValue : function(){
50566         this.value = '';
50567         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50568         
50569     },
50570
50571     /**
50572      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50573      * will be displayed in the field.  If the value does not match the data value of an existing item,
50574      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50575      * Otherwise the field will be blank (although the value will still be set).
50576      * @param {String} value The value to match
50577      */
50578     setValue : function(v){
50579         var d = this.el.dom;
50580         for (var i =0; i < d.options.length;i++) {
50581             if (v == d.options[i].value) {
50582                 d.selectedIndex = i;
50583                 this.value = v;
50584                 return;
50585             }
50586         }
50587         this.clearValue();
50588     },
50589     /**
50590      * @property {Object} the last set data for the element
50591      */
50592     
50593     lastData : false,
50594     /**
50595      * Sets the value of the field based on a object which is related to the record format for the store.
50596      * @param {Object} value the value to set as. or false on reset?
50597      */
50598     setFromData : function(o){
50599         Roo.log('setfrom data?');
50600          
50601         
50602         
50603     },
50604     // private
50605     reset : function(){
50606         this.clearValue();
50607     },
50608     // private
50609     findRecord : function(prop, value){
50610         
50611         return false;
50612     
50613         var record;
50614         if(this.store.getCount() > 0){
50615             this.store.each(function(r){
50616                 if(r.data[prop] == value){
50617                     record = r;
50618                     return false;
50619                 }
50620                 return true;
50621             });
50622         }
50623         return record;
50624     },
50625     
50626     getName: function()
50627     {
50628         // returns hidden if it's set..
50629         if (!this.rendered) {return ''};
50630         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
50631         
50632     },
50633      
50634
50635     
50636
50637     // private
50638     onEmptyResults : function(){
50639         Roo.log('empty results');
50640         //this.collapse();
50641     },
50642
50643     /**
50644      * Returns true if the dropdown list is expanded, else false.
50645      */
50646     isExpanded : function(){
50647         return false;
50648     },
50649
50650     /**
50651      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
50652      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50653      * @param {String} value The data value of the item to select
50654      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50655      * selected item if it is not currently in view (defaults to true)
50656      * @return {Boolean} True if the value matched an item in the list, else false
50657      */
50658     selectByValue : function(v, scrollIntoView){
50659         Roo.log('select By Value');
50660         return false;
50661     
50662         if(v !== undefined && v !== null){
50663             var r = this.findRecord(this.valueField || this.displayField, v);
50664             if(r){
50665                 this.select(this.store.indexOf(r), scrollIntoView);
50666                 return true;
50667             }
50668         }
50669         return false;
50670     },
50671
50672     /**
50673      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
50674      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50675      * @param {Number} index The zero-based index of the list item to select
50676      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50677      * selected item if it is not currently in view (defaults to true)
50678      */
50679     select : function(index, scrollIntoView){
50680         Roo.log('select ');
50681         return  ;
50682         
50683         this.selectedIndex = index;
50684         this.view.select(index);
50685         if(scrollIntoView !== false){
50686             var el = this.view.getNode(index);
50687             if(el){
50688                 this.innerList.scrollChildIntoView(el, false);
50689             }
50690         }
50691     },
50692
50693       
50694
50695     // private
50696     validateBlur : function(){
50697         
50698         return;
50699         
50700     },
50701
50702     // private
50703     initQuery : function(){
50704         this.doQuery(this.getRawValue());
50705     },
50706
50707     // private
50708     doForce : function(){
50709         if(this.el.dom.value.length > 0){
50710             this.el.dom.value =
50711                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
50712              
50713         }
50714     },
50715
50716     /**
50717      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
50718      * query allowing the query action to be canceled if needed.
50719      * @param {String} query The SQL query to execute
50720      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
50721      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
50722      * saved in the current store (defaults to false)
50723      */
50724     doQuery : function(q, forceAll){
50725         
50726         Roo.log('doQuery?');
50727         if(q === undefined || q === null){
50728             q = '';
50729         }
50730         var qe = {
50731             query: q,
50732             forceAll: forceAll,
50733             combo: this,
50734             cancel:false
50735         };
50736         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
50737             return false;
50738         }
50739         q = qe.query;
50740         forceAll = qe.forceAll;
50741         if(forceAll === true || (q.length >= this.minChars)){
50742             if(this.lastQuery != q || this.alwaysQuery){
50743                 this.lastQuery = q;
50744                 if(this.mode == 'local'){
50745                     this.selectedIndex = -1;
50746                     if(forceAll){
50747                         this.store.clearFilter();
50748                     }else{
50749                         this.store.filter(this.displayField, q);
50750                     }
50751                     this.onLoad();
50752                 }else{
50753                     this.store.baseParams[this.queryParam] = q;
50754                     this.store.load({
50755                         params: this.getParams(q)
50756                     });
50757                     this.expand();
50758                 }
50759             }else{
50760                 this.selectedIndex = -1;
50761                 this.onLoad();   
50762             }
50763         }
50764     },
50765
50766     // private
50767     getParams : function(q){
50768         var p = {};
50769         //p[this.queryParam] = q;
50770         if(this.pageSize){
50771             p.start = 0;
50772             p.limit = this.pageSize;
50773         }
50774         return p;
50775     },
50776
50777     /**
50778      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
50779      */
50780     collapse : function(){
50781         
50782     },
50783
50784     // private
50785     collapseIf : function(e){
50786         
50787     },
50788
50789     /**
50790      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
50791      */
50792     expand : function(){
50793         
50794     } ,
50795
50796     // private
50797      
50798
50799     /** 
50800     * @cfg {Boolean} grow 
50801     * @hide 
50802     */
50803     /** 
50804     * @cfg {Number} growMin 
50805     * @hide 
50806     */
50807     /** 
50808     * @cfg {Number} growMax 
50809     * @hide 
50810     */
50811     /**
50812      * @hide
50813      * @method autoSize
50814      */
50815     
50816     setWidth : function()
50817     {
50818         
50819     },
50820     getResizeEl : function(){
50821         return this.el;
50822     }
50823 });//<script type="text/javasscript">
50824  
50825
50826 /**
50827  * @class Roo.DDView
50828  * A DnD enabled version of Roo.View.
50829  * @param {Element/String} container The Element in which to create the View.
50830  * @param {String} tpl The template string used to create the markup for each element of the View
50831  * @param {Object} config The configuration properties. These include all the config options of
50832  * {@link Roo.View} plus some specific to this class.<br>
50833  * <p>
50834  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
50835  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
50836  * <p>
50837  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
50838 .x-view-drag-insert-above {
50839         border-top:1px dotted #3366cc;
50840 }
50841 .x-view-drag-insert-below {
50842         border-bottom:1px dotted #3366cc;
50843 }
50844 </code></pre>
50845  * 
50846  */
50847  
50848 Roo.DDView = function(container, tpl, config) {
50849     Roo.DDView.superclass.constructor.apply(this, arguments);
50850     this.getEl().setStyle("outline", "0px none");
50851     this.getEl().unselectable();
50852     if (this.dragGroup) {
50853                 this.setDraggable(this.dragGroup.split(","));
50854     }
50855     if (this.dropGroup) {
50856                 this.setDroppable(this.dropGroup.split(","));
50857     }
50858     if (this.deletable) {
50859         this.setDeletable();
50860     }
50861     this.isDirtyFlag = false;
50862         this.addEvents({
50863                 "drop" : true
50864         });
50865 };
50866
50867 Roo.extend(Roo.DDView, Roo.View, {
50868 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
50869 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
50870 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
50871 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
50872
50873         isFormField: true,
50874
50875         reset: Roo.emptyFn,
50876         
50877         clearInvalid: Roo.form.Field.prototype.clearInvalid,
50878
50879         validate: function() {
50880                 return true;
50881         },
50882         
50883         destroy: function() {
50884                 this.purgeListeners();
50885                 this.getEl.removeAllListeners();
50886                 this.getEl().remove();
50887                 if (this.dragZone) {
50888                         if (this.dragZone.destroy) {
50889                                 this.dragZone.destroy();
50890                         }
50891                 }
50892                 if (this.dropZone) {
50893                         if (this.dropZone.destroy) {
50894                                 this.dropZone.destroy();
50895                         }
50896                 }
50897         },
50898
50899 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
50900         getName: function() {
50901                 return this.name;
50902         },
50903
50904 /**     Loads the View from a JSON string representing the Records to put into the Store. */
50905         setValue: function(v) {
50906                 if (!this.store) {
50907                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
50908                 }
50909                 var data = {};
50910                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
50911                 this.store.proxy = new Roo.data.MemoryProxy(data);
50912                 this.store.load();
50913         },
50914
50915 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
50916         getValue: function() {
50917                 var result = '(';
50918                 this.store.each(function(rec) {
50919                         result += rec.id + ',';
50920                 });
50921                 return result.substr(0, result.length - 1) + ')';
50922         },
50923         
50924         getIds: function() {
50925                 var i = 0, result = new Array(this.store.getCount());
50926                 this.store.each(function(rec) {
50927                         result[i++] = rec.id;
50928                 });
50929                 return result;
50930         },
50931         
50932         isDirty: function() {
50933                 return this.isDirtyFlag;
50934         },
50935
50936 /**
50937  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
50938  *      whole Element becomes the target, and this causes the drop gesture to append.
50939  */
50940     getTargetFromEvent : function(e) {
50941                 var target = e.getTarget();
50942                 while ((target !== null) && (target.parentNode != this.el.dom)) {
50943                 target = target.parentNode;
50944                 }
50945                 if (!target) {
50946                         target = this.el.dom.lastChild || this.el.dom;
50947                 }
50948                 return target;
50949     },
50950
50951 /**
50952  *      Create the drag data which consists of an object which has the property "ddel" as
50953  *      the drag proxy element. 
50954  */
50955     getDragData : function(e) {
50956         var target = this.findItemFromChild(e.getTarget());
50957                 if(target) {
50958                         this.handleSelection(e);
50959                         var selNodes = this.getSelectedNodes();
50960             var dragData = {
50961                 source: this,
50962                 copy: this.copy || (this.allowCopy && e.ctrlKey),
50963                 nodes: selNodes,
50964                 records: []
50965                         };
50966                         var selectedIndices = this.getSelectedIndexes();
50967                         for (var i = 0; i < selectedIndices.length; i++) {
50968                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
50969                         }
50970                         if (selNodes.length == 1) {
50971                                 dragData.ddel = target.cloneNode(true); // the div element
50972                         } else {
50973                                 var div = document.createElement('div'); // create the multi element drag "ghost"
50974                                 div.className = 'multi-proxy';
50975                                 for (var i = 0, len = selNodes.length; i < len; i++) {
50976                                         div.appendChild(selNodes[i].cloneNode(true));
50977                                 }
50978                                 dragData.ddel = div;
50979                         }
50980             //console.log(dragData)
50981             //console.log(dragData.ddel.innerHTML)
50982                         return dragData;
50983                 }
50984         //console.log('nodragData')
50985                 return false;
50986     },
50987     
50988 /**     Specify to which ddGroup items in this DDView may be dragged. */
50989     setDraggable: function(ddGroup) {
50990         if (ddGroup instanceof Array) {
50991                 Roo.each(ddGroup, this.setDraggable, this);
50992                 return;
50993         }
50994         if (this.dragZone) {
50995                 this.dragZone.addToGroup(ddGroup);
50996         } else {
50997                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
50998                                 containerScroll: true,
50999                                 ddGroup: ddGroup 
51000
51001                         });
51002 //                      Draggability implies selection. DragZone's mousedown selects the element.
51003                         if (!this.multiSelect) { this.singleSelect = true; }
51004
51005 //                      Wire the DragZone's handlers up to methods in *this*
51006                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51007                 }
51008     },
51009
51010 /**     Specify from which ddGroup this DDView accepts drops. */
51011     setDroppable: function(ddGroup) {
51012         if (ddGroup instanceof Array) {
51013                 Roo.each(ddGroup, this.setDroppable, this);
51014                 return;
51015         }
51016         if (this.dropZone) {
51017                 this.dropZone.addToGroup(ddGroup);
51018         } else {
51019                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51020                                 containerScroll: true,
51021                                 ddGroup: ddGroup
51022                         });
51023
51024 //                      Wire the DropZone's handlers up to methods in *this*
51025                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51026                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51027                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51028                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51029                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51030                 }
51031     },
51032
51033 /**     Decide whether to drop above or below a View node. */
51034     getDropPoint : function(e, n, dd){
51035         if (n == this.el.dom) { return "above"; }
51036                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51037                 var c = t + (b - t) / 2;
51038                 var y = Roo.lib.Event.getPageY(e);
51039                 if(y <= c) {
51040                         return "above";
51041                 }else{
51042                         return "below";
51043                 }
51044     },
51045
51046     onNodeEnter : function(n, dd, e, data){
51047                 return false;
51048     },
51049     
51050     onNodeOver : function(n, dd, e, data){
51051                 var pt = this.getDropPoint(e, n, dd);
51052                 // set the insert point style on the target node
51053                 var dragElClass = this.dropNotAllowed;
51054                 if (pt) {
51055                         var targetElClass;
51056                         if (pt == "above"){
51057                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51058                                 targetElClass = "x-view-drag-insert-above";
51059                         } else {
51060                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51061                                 targetElClass = "x-view-drag-insert-below";
51062                         }
51063                         if (this.lastInsertClass != targetElClass){
51064                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51065                                 this.lastInsertClass = targetElClass;
51066                         }
51067                 }
51068                 return dragElClass;
51069         },
51070
51071     onNodeOut : function(n, dd, e, data){
51072                 this.removeDropIndicators(n);
51073     },
51074
51075     onNodeDrop : function(n, dd, e, data){
51076         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51077                 return false;
51078         }
51079         var pt = this.getDropPoint(e, n, dd);
51080                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51081                 if (pt == "below") { insertAt++; }
51082                 for (var i = 0; i < data.records.length; i++) {
51083                         var r = data.records[i];
51084                         var dup = this.store.getById(r.id);
51085                         if (dup && (dd != this.dragZone)) {
51086                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51087                         } else {
51088                                 if (data.copy) {
51089                                         this.store.insert(insertAt++, r.copy());
51090                                 } else {
51091                                         data.source.isDirtyFlag = true;
51092                                         r.store.remove(r);
51093                                         this.store.insert(insertAt++, r);
51094                                 }
51095                                 this.isDirtyFlag = true;
51096                         }
51097                 }
51098                 this.dragZone.cachedTarget = null;
51099                 return true;
51100     },
51101
51102     removeDropIndicators : function(n){
51103                 if(n){
51104                         Roo.fly(n).removeClass([
51105                                 "x-view-drag-insert-above",
51106                                 "x-view-drag-insert-below"]);
51107                         this.lastInsertClass = "_noclass";
51108                 }
51109     },
51110
51111 /**
51112  *      Utility method. Add a delete option to the DDView's context menu.
51113  *      @param {String} imageUrl The URL of the "delete" icon image.
51114  */
51115         setDeletable: function(imageUrl) {
51116                 if (!this.singleSelect && !this.multiSelect) {
51117                         this.singleSelect = true;
51118                 }
51119                 var c = this.getContextMenu();
51120                 this.contextMenu.on("itemclick", function(item) {
51121                         switch (item.id) {
51122                                 case "delete":
51123                                         this.remove(this.getSelectedIndexes());
51124                                         break;
51125                         }
51126                 }, this);
51127                 this.contextMenu.add({
51128                         icon: imageUrl,
51129                         id: "delete",
51130                         text: 'Delete'
51131                 });
51132         },
51133         
51134 /**     Return the context menu for this DDView. */
51135         getContextMenu: function() {
51136                 if (!this.contextMenu) {
51137 //                      Create the View's context menu
51138                         this.contextMenu = new Roo.menu.Menu({
51139                                 id: this.id + "-contextmenu"
51140                         });
51141                         this.el.on("contextmenu", this.showContextMenu, this);
51142                 }
51143                 return this.contextMenu;
51144         },
51145         
51146         disableContextMenu: function() {
51147                 if (this.contextMenu) {
51148                         this.el.un("contextmenu", this.showContextMenu, this);
51149                 }
51150         },
51151
51152         showContextMenu: function(e, item) {
51153         item = this.findItemFromChild(e.getTarget());
51154                 if (item) {
51155                         e.stopEvent();
51156                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51157                         this.contextMenu.showAt(e.getXY());
51158             }
51159     },
51160
51161 /**
51162  *      Remove {@link Roo.data.Record}s at the specified indices.
51163  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51164  */
51165     remove: function(selectedIndices) {
51166                 selectedIndices = [].concat(selectedIndices);
51167                 for (var i = 0; i < selectedIndices.length; i++) {
51168                         var rec = this.store.getAt(selectedIndices[i]);
51169                         this.store.remove(rec);
51170                 }
51171     },
51172
51173 /**
51174  *      Double click fires the event, but also, if this is draggable, and there is only one other
51175  *      related DropZone, it transfers the selected node.
51176  */
51177     onDblClick : function(e){
51178         var item = this.findItemFromChild(e.getTarget());
51179         if(item){
51180             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51181                 return false;
51182             }
51183             if (this.dragGroup) {
51184                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51185                     while (targets.indexOf(this.dropZone) > -1) {
51186                             targets.remove(this.dropZone);
51187                                 }
51188                     if (targets.length == 1) {
51189                                         this.dragZone.cachedTarget = null;
51190                         var el = Roo.get(targets[0].getEl());
51191                         var box = el.getBox(true);
51192                         targets[0].onNodeDrop(el.dom, {
51193                                 target: el.dom,
51194                                 xy: [box.x, box.y + box.height - 1]
51195                         }, null, this.getDragData(e));
51196                     }
51197                 }
51198         }
51199     },
51200     
51201     handleSelection: function(e) {
51202                 this.dragZone.cachedTarget = null;
51203         var item = this.findItemFromChild(e.getTarget());
51204         if (!item) {
51205                 this.clearSelections(true);
51206                 return;
51207         }
51208                 if (item && (this.multiSelect || this.singleSelect)){
51209                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51210                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51211                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51212                                 this.unselect(item);
51213                         } else {
51214                                 this.select(item, this.multiSelect && e.ctrlKey);
51215                                 this.lastSelection = item;
51216                         }
51217                 }
51218     },
51219
51220     onItemClick : function(item, index, e){
51221                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51222                         return false;
51223                 }
51224                 return true;
51225     },
51226
51227     unselect : function(nodeInfo, suppressEvent){
51228                 var node = this.getNode(nodeInfo);
51229                 if(node && this.isSelected(node)){
51230                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51231                                 Roo.fly(node).removeClass(this.selectedClass);
51232                                 this.selections.remove(node);
51233                                 if(!suppressEvent){
51234                                         this.fireEvent("selectionchange", this, this.selections);
51235                                 }
51236                         }
51237                 }
51238     }
51239 });
51240 /*
51241  * Based on:
51242  * Ext JS Library 1.1.1
51243  * Copyright(c) 2006-2007, Ext JS, LLC.
51244  *
51245  * Originally Released Under LGPL - original licence link has changed is not relivant.
51246  *
51247  * Fork - LGPL
51248  * <script type="text/javascript">
51249  */
51250  
51251 /**
51252  * @class Roo.LayoutManager
51253  * @extends Roo.util.Observable
51254  * Base class for layout managers.
51255  */
51256 Roo.LayoutManager = function(container, config){
51257     Roo.LayoutManager.superclass.constructor.call(this);
51258     this.el = Roo.get(container);
51259     // ie scrollbar fix
51260     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51261         document.body.scroll = "no";
51262     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51263         this.el.position('relative');
51264     }
51265     this.id = this.el.id;
51266     this.el.addClass("x-layout-container");
51267     /** false to disable window resize monitoring @type Boolean */
51268     this.monitorWindowResize = true;
51269     this.regions = {};
51270     this.addEvents({
51271         /**
51272          * @event layout
51273          * Fires when a layout is performed. 
51274          * @param {Roo.LayoutManager} this
51275          */
51276         "layout" : true,
51277         /**
51278          * @event regionresized
51279          * Fires when the user resizes a region. 
51280          * @param {Roo.LayoutRegion} region The resized region
51281          * @param {Number} newSize The new size (width for east/west, height for north/south)
51282          */
51283         "regionresized" : true,
51284         /**
51285          * @event regioncollapsed
51286          * Fires when a region is collapsed. 
51287          * @param {Roo.LayoutRegion} region The collapsed region
51288          */
51289         "regioncollapsed" : true,
51290         /**
51291          * @event regionexpanded
51292          * Fires when a region is expanded.  
51293          * @param {Roo.LayoutRegion} region The expanded region
51294          */
51295         "regionexpanded" : true
51296     });
51297     this.updating = false;
51298     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51299 };
51300
51301 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51302     /**
51303      * Returns true if this layout is currently being updated
51304      * @return {Boolean}
51305      */
51306     isUpdating : function(){
51307         return this.updating; 
51308     },
51309     
51310     /**
51311      * Suspend the LayoutManager from doing auto-layouts while
51312      * making multiple add or remove calls
51313      */
51314     beginUpdate : function(){
51315         this.updating = true;    
51316     },
51317     
51318     /**
51319      * Restore auto-layouts and optionally disable the manager from performing a layout
51320      * @param {Boolean} noLayout true to disable a layout update 
51321      */
51322     endUpdate : function(noLayout){
51323         this.updating = false;
51324         if(!noLayout){
51325             this.layout();
51326         }    
51327     },
51328     
51329     layout: function(){
51330         
51331     },
51332     
51333     onRegionResized : function(region, newSize){
51334         this.fireEvent("regionresized", region, newSize);
51335         this.layout();
51336     },
51337     
51338     onRegionCollapsed : function(region){
51339         this.fireEvent("regioncollapsed", region);
51340     },
51341     
51342     onRegionExpanded : function(region){
51343         this.fireEvent("regionexpanded", region);
51344     },
51345         
51346     /**
51347      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51348      * performs box-model adjustments.
51349      * @return {Object} The size as an object {width: (the width), height: (the height)}
51350      */
51351     getViewSize : function(){
51352         var size;
51353         if(this.el.dom != document.body){
51354             size = this.el.getSize();
51355         }else{
51356             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51357         }
51358         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51359         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51360         return size;
51361     },
51362     
51363     /**
51364      * Returns the Element this layout is bound to.
51365      * @return {Roo.Element}
51366      */
51367     getEl : function(){
51368         return this.el;
51369     },
51370     
51371     /**
51372      * Returns the specified region.
51373      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51374      * @return {Roo.LayoutRegion}
51375      */
51376     getRegion : function(target){
51377         return this.regions[target.toLowerCase()];
51378     },
51379     
51380     onWindowResize : function(){
51381         if(this.monitorWindowResize){
51382             this.layout();
51383         }
51384     }
51385 });/*
51386  * Based on:
51387  * Ext JS Library 1.1.1
51388  * Copyright(c) 2006-2007, Ext JS, LLC.
51389  *
51390  * Originally Released Under LGPL - original licence link has changed is not relivant.
51391  *
51392  * Fork - LGPL
51393  * <script type="text/javascript">
51394  */
51395 /**
51396  * @class Roo.BorderLayout
51397  * @extends Roo.LayoutManager
51398  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51399  * please see: <br><br>
51400  * <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>
51401  * <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>
51402  * Example:
51403  <pre><code>
51404  var layout = new Roo.BorderLayout(document.body, {
51405     north: {
51406         initialSize: 25,
51407         titlebar: false
51408     },
51409     west: {
51410         split:true,
51411         initialSize: 200,
51412         minSize: 175,
51413         maxSize: 400,
51414         titlebar: true,
51415         collapsible: true
51416     },
51417     east: {
51418         split:true,
51419         initialSize: 202,
51420         minSize: 175,
51421         maxSize: 400,
51422         titlebar: true,
51423         collapsible: true
51424     },
51425     south: {
51426         split:true,
51427         initialSize: 100,
51428         minSize: 100,
51429         maxSize: 200,
51430         titlebar: true,
51431         collapsible: true
51432     },
51433     center: {
51434         titlebar: true,
51435         autoScroll:true,
51436         resizeTabs: true,
51437         minTabWidth: 50,
51438         preferredTabWidth: 150
51439     }
51440 });
51441
51442 // shorthand
51443 var CP = Roo.ContentPanel;
51444
51445 layout.beginUpdate();
51446 layout.add("north", new CP("north", "North"));
51447 layout.add("south", new CP("south", {title: "South", closable: true}));
51448 layout.add("west", new CP("west", {title: "West"}));
51449 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51450 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51451 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51452 layout.getRegion("center").showPanel("center1");
51453 layout.endUpdate();
51454 </code></pre>
51455
51456 <b>The container the layout is rendered into can be either the body element or any other element.
51457 If it is not the body element, the container needs to either be an absolute positioned element,
51458 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51459 the container size if it is not the body element.</b>
51460
51461 * @constructor
51462 * Create a new BorderLayout
51463 * @param {String/HTMLElement/Element} container The container this layout is bound to
51464 * @param {Object} config Configuration options
51465  */
51466 Roo.BorderLayout = function(container, config){
51467     config = config || {};
51468     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51469     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51470     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51471         var target = this.factory.validRegions[i];
51472         if(config[target]){
51473             this.addRegion(target, config[target]);
51474         }
51475     }
51476 };
51477
51478 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51479     /**
51480      * Creates and adds a new region if it doesn't already exist.
51481      * @param {String} target The target region key (north, south, east, west or center).
51482      * @param {Object} config The regions config object
51483      * @return {BorderLayoutRegion} The new region
51484      */
51485     addRegion : function(target, config){
51486         if(!this.regions[target]){
51487             var r = this.factory.create(target, this, config);
51488             this.bindRegion(target, r);
51489         }
51490         return this.regions[target];
51491     },
51492
51493     // private (kinda)
51494     bindRegion : function(name, r){
51495         this.regions[name] = r;
51496         r.on("visibilitychange", this.layout, this);
51497         r.on("paneladded", this.layout, this);
51498         r.on("panelremoved", this.layout, this);
51499         r.on("invalidated", this.layout, this);
51500         r.on("resized", this.onRegionResized, this);
51501         r.on("collapsed", this.onRegionCollapsed, this);
51502         r.on("expanded", this.onRegionExpanded, this);
51503     },
51504
51505     /**
51506      * Performs a layout update.
51507      */
51508     layout : function(){
51509         if(this.updating) {
51510             return;
51511         }
51512         var size = this.getViewSize();
51513         var w = size.width;
51514         var h = size.height;
51515         var centerW = w;
51516         var centerH = h;
51517         var centerY = 0;
51518         var centerX = 0;
51519         //var x = 0, y = 0;
51520
51521         var rs = this.regions;
51522         var north = rs["north"];
51523         var south = rs["south"]; 
51524         var west = rs["west"];
51525         var east = rs["east"];
51526         var center = rs["center"];
51527         //if(this.hideOnLayout){ // not supported anymore
51528             //c.el.setStyle("display", "none");
51529         //}
51530         if(north && north.isVisible()){
51531             var b = north.getBox();
51532             var m = north.getMargins();
51533             b.width = w - (m.left+m.right);
51534             b.x = m.left;
51535             b.y = m.top;
51536             centerY = b.height + b.y + m.bottom;
51537             centerH -= centerY;
51538             north.updateBox(this.safeBox(b));
51539         }
51540         if(south && south.isVisible()){
51541             var b = south.getBox();
51542             var m = south.getMargins();
51543             b.width = w - (m.left+m.right);
51544             b.x = m.left;
51545             var totalHeight = (b.height + m.top + m.bottom);
51546             b.y = h - totalHeight + m.top;
51547             centerH -= totalHeight;
51548             south.updateBox(this.safeBox(b));
51549         }
51550         if(west && west.isVisible()){
51551             var b = west.getBox();
51552             var m = west.getMargins();
51553             b.height = centerH - (m.top+m.bottom);
51554             b.x = m.left;
51555             b.y = centerY + m.top;
51556             var totalWidth = (b.width + m.left + m.right);
51557             centerX += totalWidth;
51558             centerW -= totalWidth;
51559             west.updateBox(this.safeBox(b));
51560         }
51561         if(east && east.isVisible()){
51562             var b = east.getBox();
51563             var m = east.getMargins();
51564             b.height = centerH - (m.top+m.bottom);
51565             var totalWidth = (b.width + m.left + m.right);
51566             b.x = w - totalWidth + m.left;
51567             b.y = centerY + m.top;
51568             centerW -= totalWidth;
51569             east.updateBox(this.safeBox(b));
51570         }
51571         if(center){
51572             var m = center.getMargins();
51573             var centerBox = {
51574                 x: centerX + m.left,
51575                 y: centerY + m.top,
51576                 width: centerW - (m.left+m.right),
51577                 height: centerH - (m.top+m.bottom)
51578             };
51579             //if(this.hideOnLayout){
51580                 //center.el.setStyle("display", "block");
51581             //}
51582             center.updateBox(this.safeBox(centerBox));
51583         }
51584         this.el.repaint();
51585         this.fireEvent("layout", this);
51586     },
51587
51588     // private
51589     safeBox : function(box){
51590         box.width = Math.max(0, box.width);
51591         box.height = Math.max(0, box.height);
51592         return box;
51593     },
51594
51595     /**
51596      * Adds a ContentPanel (or subclass) to this layout.
51597      * @param {String} target The target region key (north, south, east, west or center).
51598      * @param {Roo.ContentPanel} panel The panel to add
51599      * @return {Roo.ContentPanel} The added panel
51600      */
51601     add : function(target, panel){
51602          
51603         target = target.toLowerCase();
51604         return this.regions[target].add(panel);
51605     },
51606
51607     /**
51608      * Remove a ContentPanel (or subclass) to this layout.
51609      * @param {String} target The target region key (north, south, east, west or center).
51610      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
51611      * @return {Roo.ContentPanel} The removed panel
51612      */
51613     remove : function(target, panel){
51614         target = target.toLowerCase();
51615         return this.regions[target].remove(panel);
51616     },
51617
51618     /**
51619      * Searches all regions for a panel with the specified id
51620      * @param {String} panelId
51621      * @return {Roo.ContentPanel} The panel or null if it wasn't found
51622      */
51623     findPanel : function(panelId){
51624         var rs = this.regions;
51625         for(var target in rs){
51626             if(typeof rs[target] != "function"){
51627                 var p = rs[target].getPanel(panelId);
51628                 if(p){
51629                     return p;
51630                 }
51631             }
51632         }
51633         return null;
51634     },
51635
51636     /**
51637      * Searches all regions for a panel with the specified id and activates (shows) it.
51638      * @param {String/ContentPanel} panelId The panels id or the panel itself
51639      * @return {Roo.ContentPanel} The shown panel or null
51640      */
51641     showPanel : function(panelId) {
51642       var rs = this.regions;
51643       for(var target in rs){
51644          var r = rs[target];
51645          if(typeof r != "function"){
51646             if(r.hasPanel(panelId)){
51647                return r.showPanel(panelId);
51648             }
51649          }
51650       }
51651       return null;
51652    },
51653
51654    /**
51655      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
51656      * @param {Roo.state.Provider} provider (optional) An alternate state provider
51657      */
51658     restoreState : function(provider){
51659         if(!provider){
51660             provider = Roo.state.Manager;
51661         }
51662         var sm = new Roo.LayoutStateManager();
51663         sm.init(this, provider);
51664     },
51665
51666     /**
51667      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
51668      * object should contain properties for each region to add ContentPanels to, and each property's value should be
51669      * a valid ContentPanel config object.  Example:
51670      * <pre><code>
51671 // Create the main layout
51672 var layout = new Roo.BorderLayout('main-ct', {
51673     west: {
51674         split:true,
51675         minSize: 175,
51676         titlebar: true
51677     },
51678     center: {
51679         title:'Components'
51680     }
51681 }, 'main-ct');
51682
51683 // Create and add multiple ContentPanels at once via configs
51684 layout.batchAdd({
51685    west: {
51686        id: 'source-files',
51687        autoCreate:true,
51688        title:'Ext Source Files',
51689        autoScroll:true,
51690        fitToFrame:true
51691    },
51692    center : {
51693        el: cview,
51694        autoScroll:true,
51695        fitToFrame:true,
51696        toolbar: tb,
51697        resizeEl:'cbody'
51698    }
51699 });
51700 </code></pre>
51701      * @param {Object} regions An object containing ContentPanel configs by region name
51702      */
51703     batchAdd : function(regions){
51704         this.beginUpdate();
51705         for(var rname in regions){
51706             var lr = this.regions[rname];
51707             if(lr){
51708                 this.addTypedPanels(lr, regions[rname]);
51709             }
51710         }
51711         this.endUpdate();
51712     },
51713
51714     // private
51715     addTypedPanels : function(lr, ps){
51716         if(typeof ps == 'string'){
51717             lr.add(new Roo.ContentPanel(ps));
51718         }
51719         else if(ps instanceof Array){
51720             for(var i =0, len = ps.length; i < len; i++){
51721                 this.addTypedPanels(lr, ps[i]);
51722             }
51723         }
51724         else if(!ps.events){ // raw config?
51725             var el = ps.el;
51726             delete ps.el; // prevent conflict
51727             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
51728         }
51729         else {  // panel object assumed!
51730             lr.add(ps);
51731         }
51732     },
51733     /**
51734      * Adds a xtype elements to the layout.
51735      * <pre><code>
51736
51737 layout.addxtype({
51738        xtype : 'ContentPanel',
51739        region: 'west',
51740        items: [ .... ]
51741    }
51742 );
51743
51744 layout.addxtype({
51745         xtype : 'NestedLayoutPanel',
51746         region: 'west',
51747         layout: {
51748            center: { },
51749            west: { }   
51750         },
51751         items : [ ... list of content panels or nested layout panels.. ]
51752    }
51753 );
51754 </code></pre>
51755      * @param {Object} cfg Xtype definition of item to add.
51756      */
51757     addxtype : function(cfg)
51758     {
51759         // basically accepts a pannel...
51760         // can accept a layout region..!?!?
51761         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
51762         
51763         if (!cfg.xtype.match(/Panel$/)) {
51764             return false;
51765         }
51766         var ret = false;
51767         
51768         if (typeof(cfg.region) == 'undefined') {
51769             Roo.log("Failed to add Panel, region was not set");
51770             Roo.log(cfg);
51771             return false;
51772         }
51773         var region = cfg.region;
51774         delete cfg.region;
51775         
51776           
51777         var xitems = [];
51778         if (cfg.items) {
51779             xitems = cfg.items;
51780             delete cfg.items;
51781         }
51782         var nb = false;
51783         
51784         switch(cfg.xtype) 
51785         {
51786             case 'ContentPanel':  // ContentPanel (el, cfg)
51787             case 'ScrollPanel':  // ContentPanel (el, cfg)
51788             case 'ViewPanel': 
51789                 if(cfg.autoCreate) {
51790                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51791                 } else {
51792                     var el = this.el.createChild();
51793                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
51794                 }
51795                 
51796                 this.add(region, ret);
51797                 break;
51798             
51799             
51800             case 'TreePanel': // our new panel!
51801                 cfg.el = this.el.createChild();
51802                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51803                 this.add(region, ret);
51804                 break;
51805             
51806             case 'NestedLayoutPanel': 
51807                 // create a new Layout (which is  a Border Layout...
51808                 var el = this.el.createChild();
51809                 var clayout = cfg.layout;
51810                 delete cfg.layout;
51811                 clayout.items   = clayout.items  || [];
51812                 // replace this exitems with the clayout ones..
51813                 xitems = clayout.items;
51814                  
51815                 
51816                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
51817                     cfg.background = false;
51818                 }
51819                 var layout = new Roo.BorderLayout(el, clayout);
51820                 
51821                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
51822                 //console.log('adding nested layout panel '  + cfg.toSource());
51823                 this.add(region, ret);
51824                 nb = {}; /// find first...
51825                 break;
51826                 
51827             case 'GridPanel': 
51828             
51829                 // needs grid and region
51830                 
51831                 //var el = this.getRegion(region).el.createChild();
51832                 var el = this.el.createChild();
51833                 // create the grid first...
51834                 
51835                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
51836                 delete cfg.grid;
51837                 if (region == 'center' && this.active ) {
51838                     cfg.background = false;
51839                 }
51840                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
51841                 
51842                 this.add(region, ret);
51843                 if (cfg.background) {
51844                     ret.on('activate', function(gp) {
51845                         if (!gp.grid.rendered) {
51846                             gp.grid.render();
51847                         }
51848                     });
51849                 } else {
51850                     grid.render();
51851                 }
51852                 break;
51853            
51854            
51855            
51856                 
51857                 
51858                 
51859             default:
51860                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
51861                     
51862                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51863                     this.add(region, ret);
51864                 } else {
51865                 
51866                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
51867                     return null;
51868                 }
51869                 
51870              // GridPanel (grid, cfg)
51871             
51872         }
51873         this.beginUpdate();
51874         // add children..
51875         var region = '';
51876         var abn = {};
51877         Roo.each(xitems, function(i)  {
51878             region = nb && i.region ? i.region : false;
51879             
51880             var add = ret.addxtype(i);
51881            
51882             if (region) {
51883                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
51884                 if (!i.background) {
51885                     abn[region] = nb[region] ;
51886                 }
51887             }
51888             
51889         });
51890         this.endUpdate();
51891
51892         // make the last non-background panel active..
51893         //if (nb) { Roo.log(abn); }
51894         if (nb) {
51895             
51896             for(var r in abn) {
51897                 region = this.getRegion(r);
51898                 if (region) {
51899                     // tried using nb[r], but it does not work..
51900                      
51901                     region.showPanel(abn[r]);
51902                    
51903                 }
51904             }
51905         }
51906         return ret;
51907         
51908     }
51909 });
51910
51911 /**
51912  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
51913  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
51914  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
51915  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
51916  * <pre><code>
51917 // shorthand
51918 var CP = Roo.ContentPanel;
51919
51920 var layout = Roo.BorderLayout.create({
51921     north: {
51922         initialSize: 25,
51923         titlebar: false,
51924         panels: [new CP("north", "North")]
51925     },
51926     west: {
51927         split:true,
51928         initialSize: 200,
51929         minSize: 175,
51930         maxSize: 400,
51931         titlebar: true,
51932         collapsible: true,
51933         panels: [new CP("west", {title: "West"})]
51934     },
51935     east: {
51936         split:true,
51937         initialSize: 202,
51938         minSize: 175,
51939         maxSize: 400,
51940         titlebar: true,
51941         collapsible: true,
51942         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
51943     },
51944     south: {
51945         split:true,
51946         initialSize: 100,
51947         minSize: 100,
51948         maxSize: 200,
51949         titlebar: true,
51950         collapsible: true,
51951         panels: [new CP("south", {title: "South", closable: true})]
51952     },
51953     center: {
51954         titlebar: true,
51955         autoScroll:true,
51956         resizeTabs: true,
51957         minTabWidth: 50,
51958         preferredTabWidth: 150,
51959         panels: [
51960             new CP("center1", {title: "Close Me", closable: true}),
51961             new CP("center2", {title: "Center Panel", closable: false})
51962         ]
51963     }
51964 }, document.body);
51965
51966 layout.getRegion("center").showPanel("center1");
51967 </code></pre>
51968  * @param config
51969  * @param targetEl
51970  */
51971 Roo.BorderLayout.create = function(config, targetEl){
51972     var layout = new Roo.BorderLayout(targetEl || document.body, config);
51973     layout.beginUpdate();
51974     var regions = Roo.BorderLayout.RegionFactory.validRegions;
51975     for(var j = 0, jlen = regions.length; j < jlen; j++){
51976         var lr = regions[j];
51977         if(layout.regions[lr] && config[lr].panels){
51978             var r = layout.regions[lr];
51979             var ps = config[lr].panels;
51980             layout.addTypedPanels(r, ps);
51981         }
51982     }
51983     layout.endUpdate();
51984     return layout;
51985 };
51986
51987 // private
51988 Roo.BorderLayout.RegionFactory = {
51989     // private
51990     validRegions : ["north","south","east","west","center"],
51991
51992     // private
51993     create : function(target, mgr, config){
51994         target = target.toLowerCase();
51995         if(config.lightweight || config.basic){
51996             return new Roo.BasicLayoutRegion(mgr, config, target);
51997         }
51998         switch(target){
51999             case "north":
52000                 return new Roo.NorthLayoutRegion(mgr, config);
52001             case "south":
52002                 return new Roo.SouthLayoutRegion(mgr, config);
52003             case "east":
52004                 return new Roo.EastLayoutRegion(mgr, config);
52005             case "west":
52006                 return new Roo.WestLayoutRegion(mgr, config);
52007             case "center":
52008                 return new Roo.CenterLayoutRegion(mgr, config);
52009         }
52010         throw 'Layout region "'+target+'" not supported.';
52011     }
52012 };/*
52013  * Based on:
52014  * Ext JS Library 1.1.1
52015  * Copyright(c) 2006-2007, Ext JS, LLC.
52016  *
52017  * Originally Released Under LGPL - original licence link has changed is not relivant.
52018  *
52019  * Fork - LGPL
52020  * <script type="text/javascript">
52021  */
52022  
52023 /**
52024  * @class Roo.BasicLayoutRegion
52025  * @extends Roo.util.Observable
52026  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52027  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52028  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52029  */
52030 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52031     this.mgr = mgr;
52032     this.position  = pos;
52033     this.events = {
52034         /**
52035          * @scope Roo.BasicLayoutRegion
52036          */
52037         
52038         /**
52039          * @event beforeremove
52040          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52041          * @param {Roo.LayoutRegion} this
52042          * @param {Roo.ContentPanel} panel The panel
52043          * @param {Object} e The cancel event object
52044          */
52045         "beforeremove" : true,
52046         /**
52047          * @event invalidated
52048          * Fires when the layout for this region is changed.
52049          * @param {Roo.LayoutRegion} this
52050          */
52051         "invalidated" : true,
52052         /**
52053          * @event visibilitychange
52054          * Fires when this region is shown or hidden 
52055          * @param {Roo.LayoutRegion} this
52056          * @param {Boolean} visibility true or false
52057          */
52058         "visibilitychange" : true,
52059         /**
52060          * @event paneladded
52061          * Fires when a panel is added. 
52062          * @param {Roo.LayoutRegion} this
52063          * @param {Roo.ContentPanel} panel The panel
52064          */
52065         "paneladded" : true,
52066         /**
52067          * @event panelremoved
52068          * Fires when a panel is removed. 
52069          * @param {Roo.LayoutRegion} this
52070          * @param {Roo.ContentPanel} panel The panel
52071          */
52072         "panelremoved" : true,
52073         /**
52074          * @event beforecollapse
52075          * Fires when this region before collapse.
52076          * @param {Roo.LayoutRegion} this
52077          */
52078         "beforecollapse" : true,
52079         /**
52080          * @event collapsed
52081          * Fires when this region is collapsed.
52082          * @param {Roo.LayoutRegion} this
52083          */
52084         "collapsed" : true,
52085         /**
52086          * @event expanded
52087          * Fires when this region is expanded.
52088          * @param {Roo.LayoutRegion} this
52089          */
52090         "expanded" : true,
52091         /**
52092          * @event slideshow
52093          * Fires when this region is slid into view.
52094          * @param {Roo.LayoutRegion} this
52095          */
52096         "slideshow" : true,
52097         /**
52098          * @event slidehide
52099          * Fires when this region slides out of view. 
52100          * @param {Roo.LayoutRegion} this
52101          */
52102         "slidehide" : true,
52103         /**
52104          * @event panelactivated
52105          * Fires when a panel is activated. 
52106          * @param {Roo.LayoutRegion} this
52107          * @param {Roo.ContentPanel} panel The activated panel
52108          */
52109         "panelactivated" : true,
52110         /**
52111          * @event resized
52112          * Fires when the user resizes this region. 
52113          * @param {Roo.LayoutRegion} this
52114          * @param {Number} newSize The new size (width for east/west, height for north/south)
52115          */
52116         "resized" : true
52117     };
52118     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52119     this.panels = new Roo.util.MixedCollection();
52120     this.panels.getKey = this.getPanelId.createDelegate(this);
52121     this.box = null;
52122     this.activePanel = null;
52123     // ensure listeners are added...
52124     
52125     if (config.listeners || config.events) {
52126         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52127             listeners : config.listeners || {},
52128             events : config.events || {}
52129         });
52130     }
52131     
52132     if(skipConfig !== true){
52133         this.applyConfig(config);
52134     }
52135 };
52136
52137 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52138     getPanelId : function(p){
52139         return p.getId();
52140     },
52141     
52142     applyConfig : function(config){
52143         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52144         this.config = config;
52145         
52146     },
52147     
52148     /**
52149      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52150      * the width, for horizontal (north, south) the height.
52151      * @param {Number} newSize The new width or height
52152      */
52153     resizeTo : function(newSize){
52154         var el = this.el ? this.el :
52155                  (this.activePanel ? this.activePanel.getEl() : null);
52156         if(el){
52157             switch(this.position){
52158                 case "east":
52159                 case "west":
52160                     el.setWidth(newSize);
52161                     this.fireEvent("resized", this, newSize);
52162                 break;
52163                 case "north":
52164                 case "south":
52165                     el.setHeight(newSize);
52166                     this.fireEvent("resized", this, newSize);
52167                 break;                
52168             }
52169         }
52170     },
52171     
52172     getBox : function(){
52173         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52174     },
52175     
52176     getMargins : function(){
52177         return this.margins;
52178     },
52179     
52180     updateBox : function(box){
52181         this.box = box;
52182         var el = this.activePanel.getEl();
52183         el.dom.style.left = box.x + "px";
52184         el.dom.style.top = box.y + "px";
52185         this.activePanel.setSize(box.width, box.height);
52186     },
52187     
52188     /**
52189      * Returns the container element for this region.
52190      * @return {Roo.Element}
52191      */
52192     getEl : function(){
52193         return this.activePanel;
52194     },
52195     
52196     /**
52197      * Returns true if this region is currently visible.
52198      * @return {Boolean}
52199      */
52200     isVisible : function(){
52201         return this.activePanel ? true : false;
52202     },
52203     
52204     setActivePanel : function(panel){
52205         panel = this.getPanel(panel);
52206         if(this.activePanel && this.activePanel != panel){
52207             this.activePanel.setActiveState(false);
52208             this.activePanel.getEl().setLeftTop(-10000,-10000);
52209         }
52210         this.activePanel = panel;
52211         panel.setActiveState(true);
52212         if(this.box){
52213             panel.setSize(this.box.width, this.box.height);
52214         }
52215         this.fireEvent("panelactivated", this, panel);
52216         this.fireEvent("invalidated");
52217     },
52218     
52219     /**
52220      * Show the specified panel.
52221      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52222      * @return {Roo.ContentPanel} The shown panel or null
52223      */
52224     showPanel : function(panel){
52225         if(panel = this.getPanel(panel)){
52226             this.setActivePanel(panel);
52227         }
52228         return panel;
52229     },
52230     
52231     /**
52232      * Get the active panel for this region.
52233      * @return {Roo.ContentPanel} The active panel or null
52234      */
52235     getActivePanel : function(){
52236         return this.activePanel;
52237     },
52238     
52239     /**
52240      * Add the passed ContentPanel(s)
52241      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52242      * @return {Roo.ContentPanel} The panel added (if only one was added)
52243      */
52244     add : function(panel){
52245         if(arguments.length > 1){
52246             for(var i = 0, len = arguments.length; i < len; i++) {
52247                 this.add(arguments[i]);
52248             }
52249             return null;
52250         }
52251         if(this.hasPanel(panel)){
52252             this.showPanel(panel);
52253             return panel;
52254         }
52255         var el = panel.getEl();
52256         if(el.dom.parentNode != this.mgr.el.dom){
52257             this.mgr.el.dom.appendChild(el.dom);
52258         }
52259         if(panel.setRegion){
52260             panel.setRegion(this);
52261         }
52262         this.panels.add(panel);
52263         el.setStyle("position", "absolute");
52264         if(!panel.background){
52265             this.setActivePanel(panel);
52266             if(this.config.initialSize && this.panels.getCount()==1){
52267                 this.resizeTo(this.config.initialSize);
52268             }
52269         }
52270         this.fireEvent("paneladded", this, panel);
52271         return panel;
52272     },
52273     
52274     /**
52275      * Returns true if the panel is in this region.
52276      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52277      * @return {Boolean}
52278      */
52279     hasPanel : function(panel){
52280         if(typeof panel == "object"){ // must be panel obj
52281             panel = panel.getId();
52282         }
52283         return this.getPanel(panel) ? true : false;
52284     },
52285     
52286     /**
52287      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52288      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52289      * @param {Boolean} preservePanel Overrides the config preservePanel option
52290      * @return {Roo.ContentPanel} The panel that was removed
52291      */
52292     remove : function(panel, preservePanel){
52293         panel = this.getPanel(panel);
52294         if(!panel){
52295             return null;
52296         }
52297         var e = {};
52298         this.fireEvent("beforeremove", this, panel, e);
52299         if(e.cancel === true){
52300             return null;
52301         }
52302         var panelId = panel.getId();
52303         this.panels.removeKey(panelId);
52304         return panel;
52305     },
52306     
52307     /**
52308      * Returns the panel specified or null if it's not in this region.
52309      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52310      * @return {Roo.ContentPanel}
52311      */
52312     getPanel : function(id){
52313         if(typeof id == "object"){ // must be panel obj
52314             return id;
52315         }
52316         return this.panels.get(id);
52317     },
52318     
52319     /**
52320      * Returns this regions position (north/south/east/west/center).
52321      * @return {String} 
52322      */
52323     getPosition: function(){
52324         return this.position;    
52325     }
52326 });/*
52327  * Based on:
52328  * Ext JS Library 1.1.1
52329  * Copyright(c) 2006-2007, Ext JS, LLC.
52330  *
52331  * Originally Released Under LGPL - original licence link has changed is not relivant.
52332  *
52333  * Fork - LGPL
52334  * <script type="text/javascript">
52335  */
52336  
52337 /**
52338  * @class Roo.LayoutRegion
52339  * @extends Roo.BasicLayoutRegion
52340  * This class represents a region in a layout manager.
52341  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52342  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52343  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52344  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52345  * @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})
52346  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52347  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52348  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52349  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52350  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52351  * @cfg {String}    title           The title for the region (overrides panel titles)
52352  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52353  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52354  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52355  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52356  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52357  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52358  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52359  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52360  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52361  * @cfg {Boolean}   showPin         True to show a pin button
52362  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52363  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52364  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52365  * @cfg {Number}    width           For East/West panels
52366  * @cfg {Number}    height          For North/South panels
52367  * @cfg {Boolean}   split           To show the splitter
52368  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52369  */
52370 Roo.LayoutRegion = function(mgr, config, pos){
52371     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52372     var dh = Roo.DomHelper;
52373     /** This region's container element 
52374     * @type Roo.Element */
52375     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52376     /** This region's title element 
52377     * @type Roo.Element */
52378
52379     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52380         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52381         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52382     ]}, true);
52383     this.titleEl.enableDisplayMode();
52384     /** This region's title text element 
52385     * @type HTMLElement */
52386     this.titleTextEl = this.titleEl.dom.firstChild;
52387     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52388     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52389     this.closeBtn.enableDisplayMode();
52390     this.closeBtn.on("click", this.closeClicked, this);
52391     this.closeBtn.hide();
52392
52393     this.createBody(config);
52394     this.visible = true;
52395     this.collapsed = false;
52396
52397     if(config.hideWhenEmpty){
52398         this.hide();
52399         this.on("paneladded", this.validateVisibility, this);
52400         this.on("panelremoved", this.validateVisibility, this);
52401     }
52402     this.applyConfig(config);
52403 };
52404
52405 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52406
52407     createBody : function(){
52408         /** This region's body element 
52409         * @type Roo.Element */
52410         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52411     },
52412
52413     applyConfig : function(c){
52414         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52415             var dh = Roo.DomHelper;
52416             if(c.titlebar !== false){
52417                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52418                 this.collapseBtn.on("click", this.collapse, this);
52419                 this.collapseBtn.enableDisplayMode();
52420
52421                 if(c.showPin === true || this.showPin){
52422                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52423                     this.stickBtn.enableDisplayMode();
52424                     this.stickBtn.on("click", this.expand, this);
52425                     this.stickBtn.hide();
52426                 }
52427             }
52428             /** This region's collapsed element
52429             * @type Roo.Element */
52430             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52431                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52432             ]}, true);
52433             if(c.floatable !== false){
52434                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52435                this.collapsedEl.on("click", this.collapseClick, this);
52436             }
52437
52438             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52439                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52440                    id: "message", unselectable: "on", style:{"float":"left"}});
52441                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52442              }
52443             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52444             this.expandBtn.on("click", this.expand, this);
52445         }
52446         if(this.collapseBtn){
52447             this.collapseBtn.setVisible(c.collapsible == true);
52448         }
52449         this.cmargins = c.cmargins || this.cmargins ||
52450                          (this.position == "west" || this.position == "east" ?
52451                              {top: 0, left: 2, right:2, bottom: 0} :
52452                              {top: 2, left: 0, right:0, bottom: 2});
52453         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52454         this.bottomTabs = c.tabPosition != "top";
52455         this.autoScroll = c.autoScroll || false;
52456         if(this.autoScroll){
52457             this.bodyEl.setStyle("overflow", "auto");
52458         }else{
52459             this.bodyEl.setStyle("overflow", "hidden");
52460         }
52461         //if(c.titlebar !== false){
52462             if((!c.titlebar && !c.title) || c.titlebar === false){
52463                 this.titleEl.hide();
52464             }else{
52465                 this.titleEl.show();
52466                 if(c.title){
52467                     this.titleTextEl.innerHTML = c.title;
52468                 }
52469             }
52470         //}
52471         this.duration = c.duration || .30;
52472         this.slideDuration = c.slideDuration || .45;
52473         this.config = c;
52474         if(c.collapsed){
52475             this.collapse(true);
52476         }
52477         if(c.hidden){
52478             this.hide();
52479         }
52480     },
52481     /**
52482      * Returns true if this region is currently visible.
52483      * @return {Boolean}
52484      */
52485     isVisible : function(){
52486         return this.visible;
52487     },
52488
52489     /**
52490      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52491      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52492      */
52493     setCollapsedTitle : function(title){
52494         title = title || "&#160;";
52495         if(this.collapsedTitleTextEl){
52496             this.collapsedTitleTextEl.innerHTML = title;
52497         }
52498     },
52499
52500     getBox : function(){
52501         var b;
52502         if(!this.collapsed){
52503             b = this.el.getBox(false, true);
52504         }else{
52505             b = this.collapsedEl.getBox(false, true);
52506         }
52507         return b;
52508     },
52509
52510     getMargins : function(){
52511         return this.collapsed ? this.cmargins : this.margins;
52512     },
52513
52514     highlight : function(){
52515         this.el.addClass("x-layout-panel-dragover");
52516     },
52517
52518     unhighlight : function(){
52519         this.el.removeClass("x-layout-panel-dragover");
52520     },
52521
52522     updateBox : function(box){
52523         this.box = box;
52524         if(!this.collapsed){
52525             this.el.dom.style.left = box.x + "px";
52526             this.el.dom.style.top = box.y + "px";
52527             this.updateBody(box.width, box.height);
52528         }else{
52529             this.collapsedEl.dom.style.left = box.x + "px";
52530             this.collapsedEl.dom.style.top = box.y + "px";
52531             this.collapsedEl.setSize(box.width, box.height);
52532         }
52533         if(this.tabs){
52534             this.tabs.autoSizeTabs();
52535         }
52536     },
52537
52538     updateBody : function(w, h){
52539         if(w !== null){
52540             this.el.setWidth(w);
52541             w -= this.el.getBorderWidth("rl");
52542             if(this.config.adjustments){
52543                 w += this.config.adjustments[0];
52544             }
52545         }
52546         if(h !== null){
52547             this.el.setHeight(h);
52548             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52549             h -= this.el.getBorderWidth("tb");
52550             if(this.config.adjustments){
52551                 h += this.config.adjustments[1];
52552             }
52553             this.bodyEl.setHeight(h);
52554             if(this.tabs){
52555                 h = this.tabs.syncHeight(h);
52556             }
52557         }
52558         if(this.panelSize){
52559             w = w !== null ? w : this.panelSize.width;
52560             h = h !== null ? h : this.panelSize.height;
52561         }
52562         if(this.activePanel){
52563             var el = this.activePanel.getEl();
52564             w = w !== null ? w : el.getWidth();
52565             h = h !== null ? h : el.getHeight();
52566             this.panelSize = {width: w, height: h};
52567             this.activePanel.setSize(w, h);
52568         }
52569         if(Roo.isIE && this.tabs){
52570             this.tabs.el.repaint();
52571         }
52572     },
52573
52574     /**
52575      * Returns the container element for this region.
52576      * @return {Roo.Element}
52577      */
52578     getEl : function(){
52579         return this.el;
52580     },
52581
52582     /**
52583      * Hides this region.
52584      */
52585     hide : function(){
52586         if(!this.collapsed){
52587             this.el.dom.style.left = "-2000px";
52588             this.el.hide();
52589         }else{
52590             this.collapsedEl.dom.style.left = "-2000px";
52591             this.collapsedEl.hide();
52592         }
52593         this.visible = false;
52594         this.fireEvent("visibilitychange", this, false);
52595     },
52596
52597     /**
52598      * Shows this region if it was previously hidden.
52599      */
52600     show : function(){
52601         if(!this.collapsed){
52602             this.el.show();
52603         }else{
52604             this.collapsedEl.show();
52605         }
52606         this.visible = true;
52607         this.fireEvent("visibilitychange", this, true);
52608     },
52609
52610     closeClicked : function(){
52611         if(this.activePanel){
52612             this.remove(this.activePanel);
52613         }
52614     },
52615
52616     collapseClick : function(e){
52617         if(this.isSlid){
52618            e.stopPropagation();
52619            this.slideIn();
52620         }else{
52621            e.stopPropagation();
52622            this.slideOut();
52623         }
52624     },
52625
52626     /**
52627      * Collapses this region.
52628      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
52629      */
52630     collapse : function(skipAnim, skipCheck){
52631         if(this.collapsed) {
52632             return;
52633         }
52634         
52635         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
52636             
52637             this.collapsed = true;
52638             if(this.split){
52639                 this.split.el.hide();
52640             }
52641             if(this.config.animate && skipAnim !== true){
52642                 this.fireEvent("invalidated", this);
52643                 this.animateCollapse();
52644             }else{
52645                 this.el.setLocation(-20000,-20000);
52646                 this.el.hide();
52647                 this.collapsedEl.show();
52648                 this.fireEvent("collapsed", this);
52649                 this.fireEvent("invalidated", this);
52650             }
52651         }
52652         
52653     },
52654
52655     animateCollapse : function(){
52656         // overridden
52657     },
52658
52659     /**
52660      * Expands this region if it was previously collapsed.
52661      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
52662      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
52663      */
52664     expand : function(e, skipAnim){
52665         if(e) {
52666             e.stopPropagation();
52667         }
52668         if(!this.collapsed || this.el.hasActiveFx()) {
52669             return;
52670         }
52671         if(this.isSlid){
52672             this.afterSlideIn();
52673             skipAnim = true;
52674         }
52675         this.collapsed = false;
52676         if(this.config.animate && skipAnim !== true){
52677             this.animateExpand();
52678         }else{
52679             this.el.show();
52680             if(this.split){
52681                 this.split.el.show();
52682             }
52683             this.collapsedEl.setLocation(-2000,-2000);
52684             this.collapsedEl.hide();
52685             this.fireEvent("invalidated", this);
52686             this.fireEvent("expanded", this);
52687         }
52688     },
52689
52690     animateExpand : function(){
52691         // overridden
52692     },
52693
52694     initTabs : function()
52695     {
52696         this.bodyEl.setStyle("overflow", "hidden");
52697         var ts = new Roo.TabPanel(
52698                 this.bodyEl.dom,
52699                 {
52700                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
52701                     disableTooltips: this.config.disableTabTips,
52702                     toolbar : this.config.toolbar
52703                 }
52704         );
52705         if(this.config.hideTabs){
52706             ts.stripWrap.setDisplayed(false);
52707         }
52708         this.tabs = ts;
52709         ts.resizeTabs = this.config.resizeTabs === true;
52710         ts.minTabWidth = this.config.minTabWidth || 40;
52711         ts.maxTabWidth = this.config.maxTabWidth || 250;
52712         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
52713         ts.monitorResize = false;
52714         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52715         ts.bodyEl.addClass('x-layout-tabs-body');
52716         this.panels.each(this.initPanelAsTab, this);
52717     },
52718
52719     initPanelAsTab : function(panel){
52720         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
52721                     this.config.closeOnTab && panel.isClosable());
52722         if(panel.tabTip !== undefined){
52723             ti.setTooltip(panel.tabTip);
52724         }
52725         ti.on("activate", function(){
52726               this.setActivePanel(panel);
52727         }, this);
52728         if(this.config.closeOnTab){
52729             ti.on("beforeclose", function(t, e){
52730                 e.cancel = true;
52731                 this.remove(panel);
52732             }, this);
52733         }
52734         return ti;
52735     },
52736
52737     updatePanelTitle : function(panel, title){
52738         if(this.activePanel == panel){
52739             this.updateTitle(title);
52740         }
52741         if(this.tabs){
52742             var ti = this.tabs.getTab(panel.getEl().id);
52743             ti.setText(title);
52744             if(panel.tabTip !== undefined){
52745                 ti.setTooltip(panel.tabTip);
52746             }
52747         }
52748     },
52749
52750     updateTitle : function(title){
52751         if(this.titleTextEl && !this.config.title){
52752             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
52753         }
52754     },
52755
52756     setActivePanel : function(panel){
52757         panel = this.getPanel(panel);
52758         if(this.activePanel && this.activePanel != panel){
52759             this.activePanel.setActiveState(false);
52760         }
52761         this.activePanel = panel;
52762         panel.setActiveState(true);
52763         if(this.panelSize){
52764             panel.setSize(this.panelSize.width, this.panelSize.height);
52765         }
52766         if(this.closeBtn){
52767             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
52768         }
52769         this.updateTitle(panel.getTitle());
52770         if(this.tabs){
52771             this.fireEvent("invalidated", this);
52772         }
52773         this.fireEvent("panelactivated", this, panel);
52774     },
52775
52776     /**
52777      * Shows the specified panel.
52778      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
52779      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
52780      */
52781     showPanel : function(panel)
52782     {
52783         panel = this.getPanel(panel);
52784         if(panel){
52785             if(this.tabs){
52786                 var tab = this.tabs.getTab(panel.getEl().id);
52787                 if(tab.isHidden()){
52788                     this.tabs.unhideTab(tab.id);
52789                 }
52790                 tab.activate();
52791             }else{
52792                 this.setActivePanel(panel);
52793             }
52794         }
52795         return panel;
52796     },
52797
52798     /**
52799      * Get the active panel for this region.
52800      * @return {Roo.ContentPanel} The active panel or null
52801      */
52802     getActivePanel : function(){
52803         return this.activePanel;
52804     },
52805
52806     validateVisibility : function(){
52807         if(this.panels.getCount() < 1){
52808             this.updateTitle("&#160;");
52809             this.closeBtn.hide();
52810             this.hide();
52811         }else{
52812             if(!this.isVisible()){
52813                 this.show();
52814             }
52815         }
52816     },
52817
52818     /**
52819      * Adds the passed ContentPanel(s) to this region.
52820      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52821      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
52822      */
52823     add : function(panel){
52824         if(arguments.length > 1){
52825             for(var i = 0, len = arguments.length; i < len; i++) {
52826                 this.add(arguments[i]);
52827             }
52828             return null;
52829         }
52830         if(this.hasPanel(panel)){
52831             this.showPanel(panel);
52832             return panel;
52833         }
52834         panel.setRegion(this);
52835         this.panels.add(panel);
52836         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
52837             this.bodyEl.dom.appendChild(panel.getEl().dom);
52838             if(panel.background !== true){
52839                 this.setActivePanel(panel);
52840             }
52841             this.fireEvent("paneladded", this, panel);
52842             return panel;
52843         }
52844         if(!this.tabs){
52845             this.initTabs();
52846         }else{
52847             this.initPanelAsTab(panel);
52848         }
52849         if(panel.background !== true){
52850             this.tabs.activate(panel.getEl().id);
52851         }
52852         this.fireEvent("paneladded", this, panel);
52853         return panel;
52854     },
52855
52856     /**
52857      * Hides the tab for the specified panel.
52858      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52859      */
52860     hidePanel : function(panel){
52861         if(this.tabs && (panel = this.getPanel(panel))){
52862             this.tabs.hideTab(panel.getEl().id);
52863         }
52864     },
52865
52866     /**
52867      * Unhides the tab for a previously hidden panel.
52868      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52869      */
52870     unhidePanel : function(panel){
52871         if(this.tabs && (panel = this.getPanel(panel))){
52872             this.tabs.unhideTab(panel.getEl().id);
52873         }
52874     },
52875
52876     clearPanels : function(){
52877         while(this.panels.getCount() > 0){
52878              this.remove(this.panels.first());
52879         }
52880     },
52881
52882     /**
52883      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52884      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52885      * @param {Boolean} preservePanel Overrides the config preservePanel option
52886      * @return {Roo.ContentPanel} The panel that was removed
52887      */
52888     remove : function(panel, preservePanel){
52889         panel = this.getPanel(panel);
52890         if(!panel){
52891             return null;
52892         }
52893         var e = {};
52894         this.fireEvent("beforeremove", this, panel, e);
52895         if(e.cancel === true){
52896             return null;
52897         }
52898         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
52899         var panelId = panel.getId();
52900         this.panels.removeKey(panelId);
52901         if(preservePanel){
52902             document.body.appendChild(panel.getEl().dom);
52903         }
52904         if(this.tabs){
52905             this.tabs.removeTab(panel.getEl().id);
52906         }else if (!preservePanel){
52907             this.bodyEl.dom.removeChild(panel.getEl().dom);
52908         }
52909         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
52910             var p = this.panels.first();
52911             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
52912             tempEl.appendChild(p.getEl().dom);
52913             this.bodyEl.update("");
52914             this.bodyEl.dom.appendChild(p.getEl().dom);
52915             tempEl = null;
52916             this.updateTitle(p.getTitle());
52917             this.tabs = null;
52918             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52919             this.setActivePanel(p);
52920         }
52921         panel.setRegion(null);
52922         if(this.activePanel == panel){
52923             this.activePanel = null;
52924         }
52925         if(this.config.autoDestroy !== false && preservePanel !== true){
52926             try{panel.destroy();}catch(e){}
52927         }
52928         this.fireEvent("panelremoved", this, panel);
52929         return panel;
52930     },
52931
52932     /**
52933      * Returns the TabPanel component used by this region
52934      * @return {Roo.TabPanel}
52935      */
52936     getTabs : function(){
52937         return this.tabs;
52938     },
52939
52940     createTool : function(parentEl, className){
52941         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
52942             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
52943         btn.addClassOnOver("x-layout-tools-button-over");
52944         return btn;
52945     }
52946 });/*
52947  * Based on:
52948  * Ext JS Library 1.1.1
52949  * Copyright(c) 2006-2007, Ext JS, LLC.
52950  *
52951  * Originally Released Under LGPL - original licence link has changed is not relivant.
52952  *
52953  * Fork - LGPL
52954  * <script type="text/javascript">
52955  */
52956  
52957
52958
52959 /**
52960  * @class Roo.SplitLayoutRegion
52961  * @extends Roo.LayoutRegion
52962  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
52963  */
52964 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
52965     this.cursor = cursor;
52966     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
52967 };
52968
52969 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
52970     splitTip : "Drag to resize.",
52971     collapsibleSplitTip : "Drag to resize. Double click to hide.",
52972     useSplitTips : false,
52973
52974     applyConfig : function(config){
52975         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
52976         if(config.split){
52977             if(!this.split){
52978                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
52979                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
52980                 /** The SplitBar for this region 
52981                 * @type Roo.SplitBar */
52982                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
52983                 this.split.on("moved", this.onSplitMove, this);
52984                 this.split.useShim = config.useShim === true;
52985                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
52986                 if(this.useSplitTips){
52987                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
52988                 }
52989                 if(config.collapsible){
52990                     this.split.el.on("dblclick", this.collapse,  this);
52991                 }
52992             }
52993             if(typeof config.minSize != "undefined"){
52994                 this.split.minSize = config.minSize;
52995             }
52996             if(typeof config.maxSize != "undefined"){
52997                 this.split.maxSize = config.maxSize;
52998             }
52999             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53000                 this.hideSplitter();
53001             }
53002         }
53003     },
53004
53005     getHMaxSize : function(){
53006          var cmax = this.config.maxSize || 10000;
53007          var center = this.mgr.getRegion("center");
53008          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53009     },
53010
53011     getVMaxSize : function(){
53012          var cmax = this.config.maxSize || 10000;
53013          var center = this.mgr.getRegion("center");
53014          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53015     },
53016
53017     onSplitMove : function(split, newSize){
53018         this.fireEvent("resized", this, newSize);
53019     },
53020     
53021     /** 
53022      * Returns the {@link Roo.SplitBar} for this region.
53023      * @return {Roo.SplitBar}
53024      */
53025     getSplitBar : function(){
53026         return this.split;
53027     },
53028     
53029     hide : function(){
53030         this.hideSplitter();
53031         Roo.SplitLayoutRegion.superclass.hide.call(this);
53032     },
53033
53034     hideSplitter : function(){
53035         if(this.split){
53036             this.split.el.setLocation(-2000,-2000);
53037             this.split.el.hide();
53038         }
53039     },
53040
53041     show : function(){
53042         if(this.split){
53043             this.split.el.show();
53044         }
53045         Roo.SplitLayoutRegion.superclass.show.call(this);
53046     },
53047     
53048     beforeSlide: function(){
53049         if(Roo.isGecko){// firefox overflow auto bug workaround
53050             this.bodyEl.clip();
53051             if(this.tabs) {
53052                 this.tabs.bodyEl.clip();
53053             }
53054             if(this.activePanel){
53055                 this.activePanel.getEl().clip();
53056                 
53057                 if(this.activePanel.beforeSlide){
53058                     this.activePanel.beforeSlide();
53059                 }
53060             }
53061         }
53062     },
53063     
53064     afterSlide : function(){
53065         if(Roo.isGecko){// firefox overflow auto bug workaround
53066             this.bodyEl.unclip();
53067             if(this.tabs) {
53068                 this.tabs.bodyEl.unclip();
53069             }
53070             if(this.activePanel){
53071                 this.activePanel.getEl().unclip();
53072                 if(this.activePanel.afterSlide){
53073                     this.activePanel.afterSlide();
53074                 }
53075             }
53076         }
53077     },
53078
53079     initAutoHide : function(){
53080         if(this.autoHide !== false){
53081             if(!this.autoHideHd){
53082                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53083                 this.autoHideHd = {
53084                     "mouseout": function(e){
53085                         if(!e.within(this.el, true)){
53086                             st.delay(500);
53087                         }
53088                     },
53089                     "mouseover" : function(e){
53090                         st.cancel();
53091                     },
53092                     scope : this
53093                 };
53094             }
53095             this.el.on(this.autoHideHd);
53096         }
53097     },
53098
53099     clearAutoHide : function(){
53100         if(this.autoHide !== false){
53101             this.el.un("mouseout", this.autoHideHd.mouseout);
53102             this.el.un("mouseover", this.autoHideHd.mouseover);
53103         }
53104     },
53105
53106     clearMonitor : function(){
53107         Roo.get(document).un("click", this.slideInIf, this);
53108     },
53109
53110     // these names are backwards but not changed for compat
53111     slideOut : function(){
53112         if(this.isSlid || this.el.hasActiveFx()){
53113             return;
53114         }
53115         this.isSlid = true;
53116         if(this.collapseBtn){
53117             this.collapseBtn.hide();
53118         }
53119         this.closeBtnState = this.closeBtn.getStyle('display');
53120         this.closeBtn.hide();
53121         if(this.stickBtn){
53122             this.stickBtn.show();
53123         }
53124         this.el.show();
53125         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53126         this.beforeSlide();
53127         this.el.setStyle("z-index", 10001);
53128         this.el.slideIn(this.getSlideAnchor(), {
53129             callback: function(){
53130                 this.afterSlide();
53131                 this.initAutoHide();
53132                 Roo.get(document).on("click", this.slideInIf, this);
53133                 this.fireEvent("slideshow", this);
53134             },
53135             scope: this,
53136             block: true
53137         });
53138     },
53139
53140     afterSlideIn : function(){
53141         this.clearAutoHide();
53142         this.isSlid = false;
53143         this.clearMonitor();
53144         this.el.setStyle("z-index", "");
53145         if(this.collapseBtn){
53146             this.collapseBtn.show();
53147         }
53148         this.closeBtn.setStyle('display', this.closeBtnState);
53149         if(this.stickBtn){
53150             this.stickBtn.hide();
53151         }
53152         this.fireEvent("slidehide", this);
53153     },
53154
53155     slideIn : function(cb){
53156         if(!this.isSlid || this.el.hasActiveFx()){
53157             Roo.callback(cb);
53158             return;
53159         }
53160         this.isSlid = false;
53161         this.beforeSlide();
53162         this.el.slideOut(this.getSlideAnchor(), {
53163             callback: function(){
53164                 this.el.setLeftTop(-10000, -10000);
53165                 this.afterSlide();
53166                 this.afterSlideIn();
53167                 Roo.callback(cb);
53168             },
53169             scope: this,
53170             block: true
53171         });
53172     },
53173     
53174     slideInIf : function(e){
53175         if(!e.within(this.el)){
53176             this.slideIn();
53177         }
53178     },
53179
53180     animateCollapse : function(){
53181         this.beforeSlide();
53182         this.el.setStyle("z-index", 20000);
53183         var anchor = this.getSlideAnchor();
53184         this.el.slideOut(anchor, {
53185             callback : function(){
53186                 this.el.setStyle("z-index", "");
53187                 this.collapsedEl.slideIn(anchor, {duration:.3});
53188                 this.afterSlide();
53189                 this.el.setLocation(-10000,-10000);
53190                 this.el.hide();
53191                 this.fireEvent("collapsed", this);
53192             },
53193             scope: this,
53194             block: true
53195         });
53196     },
53197
53198     animateExpand : function(){
53199         this.beforeSlide();
53200         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53201         this.el.setStyle("z-index", 20000);
53202         this.collapsedEl.hide({
53203             duration:.1
53204         });
53205         this.el.slideIn(this.getSlideAnchor(), {
53206             callback : function(){
53207                 this.el.setStyle("z-index", "");
53208                 this.afterSlide();
53209                 if(this.split){
53210                     this.split.el.show();
53211                 }
53212                 this.fireEvent("invalidated", this);
53213                 this.fireEvent("expanded", this);
53214             },
53215             scope: this,
53216             block: true
53217         });
53218     },
53219
53220     anchors : {
53221         "west" : "left",
53222         "east" : "right",
53223         "north" : "top",
53224         "south" : "bottom"
53225     },
53226
53227     sanchors : {
53228         "west" : "l",
53229         "east" : "r",
53230         "north" : "t",
53231         "south" : "b"
53232     },
53233
53234     canchors : {
53235         "west" : "tl-tr",
53236         "east" : "tr-tl",
53237         "north" : "tl-bl",
53238         "south" : "bl-tl"
53239     },
53240
53241     getAnchor : function(){
53242         return this.anchors[this.position];
53243     },
53244
53245     getCollapseAnchor : function(){
53246         return this.canchors[this.position];
53247     },
53248
53249     getSlideAnchor : function(){
53250         return this.sanchors[this.position];
53251     },
53252
53253     getAlignAdj : function(){
53254         var cm = this.cmargins;
53255         switch(this.position){
53256             case "west":
53257                 return [0, 0];
53258             break;
53259             case "east":
53260                 return [0, 0];
53261             break;
53262             case "north":
53263                 return [0, 0];
53264             break;
53265             case "south":
53266                 return [0, 0];
53267             break;
53268         }
53269     },
53270
53271     getExpandAdj : function(){
53272         var c = this.collapsedEl, cm = this.cmargins;
53273         switch(this.position){
53274             case "west":
53275                 return [-(cm.right+c.getWidth()+cm.left), 0];
53276             break;
53277             case "east":
53278                 return [cm.right+c.getWidth()+cm.left, 0];
53279             break;
53280             case "north":
53281                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53282             break;
53283             case "south":
53284                 return [0, cm.top+cm.bottom+c.getHeight()];
53285             break;
53286         }
53287     }
53288 });/*
53289  * Based on:
53290  * Ext JS Library 1.1.1
53291  * Copyright(c) 2006-2007, Ext JS, LLC.
53292  *
53293  * Originally Released Under LGPL - original licence link has changed is not relivant.
53294  *
53295  * Fork - LGPL
53296  * <script type="text/javascript">
53297  */
53298 /*
53299  * These classes are private internal classes
53300  */
53301 Roo.CenterLayoutRegion = function(mgr, config){
53302     Roo.LayoutRegion.call(this, mgr, config, "center");
53303     this.visible = true;
53304     this.minWidth = config.minWidth || 20;
53305     this.minHeight = config.minHeight || 20;
53306 };
53307
53308 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53309     hide : function(){
53310         // center panel can't be hidden
53311     },
53312     
53313     show : function(){
53314         // center panel can't be hidden
53315     },
53316     
53317     getMinWidth: function(){
53318         return this.minWidth;
53319     },
53320     
53321     getMinHeight: function(){
53322         return this.minHeight;
53323     }
53324 });
53325
53326
53327 Roo.NorthLayoutRegion = function(mgr, config){
53328     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53329     if(this.split){
53330         this.split.placement = Roo.SplitBar.TOP;
53331         this.split.orientation = Roo.SplitBar.VERTICAL;
53332         this.split.el.addClass("x-layout-split-v");
53333     }
53334     var size = config.initialSize || config.height;
53335     if(typeof size != "undefined"){
53336         this.el.setHeight(size);
53337     }
53338 };
53339 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53340     orientation: Roo.SplitBar.VERTICAL,
53341     getBox : function(){
53342         if(this.collapsed){
53343             return this.collapsedEl.getBox();
53344         }
53345         var box = this.el.getBox();
53346         if(this.split){
53347             box.height += this.split.el.getHeight();
53348         }
53349         return box;
53350     },
53351     
53352     updateBox : function(box){
53353         if(this.split && !this.collapsed){
53354             box.height -= this.split.el.getHeight();
53355             this.split.el.setLeft(box.x);
53356             this.split.el.setTop(box.y+box.height);
53357             this.split.el.setWidth(box.width);
53358         }
53359         if(this.collapsed){
53360             this.updateBody(box.width, null);
53361         }
53362         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53363     }
53364 });
53365
53366 Roo.SouthLayoutRegion = function(mgr, config){
53367     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53368     if(this.split){
53369         this.split.placement = Roo.SplitBar.BOTTOM;
53370         this.split.orientation = Roo.SplitBar.VERTICAL;
53371         this.split.el.addClass("x-layout-split-v");
53372     }
53373     var size = config.initialSize || config.height;
53374     if(typeof size != "undefined"){
53375         this.el.setHeight(size);
53376     }
53377 };
53378 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53379     orientation: Roo.SplitBar.VERTICAL,
53380     getBox : function(){
53381         if(this.collapsed){
53382             return this.collapsedEl.getBox();
53383         }
53384         var box = this.el.getBox();
53385         if(this.split){
53386             var sh = this.split.el.getHeight();
53387             box.height += sh;
53388             box.y -= sh;
53389         }
53390         return box;
53391     },
53392     
53393     updateBox : function(box){
53394         if(this.split && !this.collapsed){
53395             var sh = this.split.el.getHeight();
53396             box.height -= sh;
53397             box.y += sh;
53398             this.split.el.setLeft(box.x);
53399             this.split.el.setTop(box.y-sh);
53400             this.split.el.setWidth(box.width);
53401         }
53402         if(this.collapsed){
53403             this.updateBody(box.width, null);
53404         }
53405         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53406     }
53407 });
53408
53409 Roo.EastLayoutRegion = function(mgr, config){
53410     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53411     if(this.split){
53412         this.split.placement = Roo.SplitBar.RIGHT;
53413         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53414         this.split.el.addClass("x-layout-split-h");
53415     }
53416     var size = config.initialSize || config.width;
53417     if(typeof size != "undefined"){
53418         this.el.setWidth(size);
53419     }
53420 };
53421 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53422     orientation: Roo.SplitBar.HORIZONTAL,
53423     getBox : function(){
53424         if(this.collapsed){
53425             return this.collapsedEl.getBox();
53426         }
53427         var box = this.el.getBox();
53428         if(this.split){
53429             var sw = this.split.el.getWidth();
53430             box.width += sw;
53431             box.x -= sw;
53432         }
53433         return box;
53434     },
53435
53436     updateBox : function(box){
53437         if(this.split && !this.collapsed){
53438             var sw = this.split.el.getWidth();
53439             box.width -= sw;
53440             this.split.el.setLeft(box.x);
53441             this.split.el.setTop(box.y);
53442             this.split.el.setHeight(box.height);
53443             box.x += sw;
53444         }
53445         if(this.collapsed){
53446             this.updateBody(null, box.height);
53447         }
53448         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53449     }
53450 });
53451
53452 Roo.WestLayoutRegion = function(mgr, config){
53453     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53454     if(this.split){
53455         this.split.placement = Roo.SplitBar.LEFT;
53456         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53457         this.split.el.addClass("x-layout-split-h");
53458     }
53459     var size = config.initialSize || config.width;
53460     if(typeof size != "undefined"){
53461         this.el.setWidth(size);
53462     }
53463 };
53464 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53465     orientation: Roo.SplitBar.HORIZONTAL,
53466     getBox : function(){
53467         if(this.collapsed){
53468             return this.collapsedEl.getBox();
53469         }
53470         var box = this.el.getBox();
53471         if(this.split){
53472             box.width += this.split.el.getWidth();
53473         }
53474         return box;
53475     },
53476     
53477     updateBox : function(box){
53478         if(this.split && !this.collapsed){
53479             var sw = this.split.el.getWidth();
53480             box.width -= sw;
53481             this.split.el.setLeft(box.x+box.width);
53482             this.split.el.setTop(box.y);
53483             this.split.el.setHeight(box.height);
53484         }
53485         if(this.collapsed){
53486             this.updateBody(null, box.height);
53487         }
53488         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53489     }
53490 });
53491 /*
53492  * Based on:
53493  * Ext JS Library 1.1.1
53494  * Copyright(c) 2006-2007, Ext JS, LLC.
53495  *
53496  * Originally Released Under LGPL - original licence link has changed is not relivant.
53497  *
53498  * Fork - LGPL
53499  * <script type="text/javascript">
53500  */
53501  
53502  
53503 /*
53504  * Private internal class for reading and applying state
53505  */
53506 Roo.LayoutStateManager = function(layout){
53507      // default empty state
53508      this.state = {
53509         north: {},
53510         south: {},
53511         east: {},
53512         west: {}       
53513     };
53514 };
53515
53516 Roo.LayoutStateManager.prototype = {
53517     init : function(layout, provider){
53518         this.provider = provider;
53519         var state = provider.get(layout.id+"-layout-state");
53520         if(state){
53521             var wasUpdating = layout.isUpdating();
53522             if(!wasUpdating){
53523                 layout.beginUpdate();
53524             }
53525             for(var key in state){
53526                 if(typeof state[key] != "function"){
53527                     var rstate = state[key];
53528                     var r = layout.getRegion(key);
53529                     if(r && rstate){
53530                         if(rstate.size){
53531                             r.resizeTo(rstate.size);
53532                         }
53533                         if(rstate.collapsed == true){
53534                             r.collapse(true);
53535                         }else{
53536                             r.expand(null, true);
53537                         }
53538                     }
53539                 }
53540             }
53541             if(!wasUpdating){
53542                 layout.endUpdate();
53543             }
53544             this.state = state; 
53545         }
53546         this.layout = layout;
53547         layout.on("regionresized", this.onRegionResized, this);
53548         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53549         layout.on("regionexpanded", this.onRegionExpanded, this);
53550     },
53551     
53552     storeState : function(){
53553         this.provider.set(this.layout.id+"-layout-state", this.state);
53554     },
53555     
53556     onRegionResized : function(region, newSize){
53557         this.state[region.getPosition()].size = newSize;
53558         this.storeState();
53559     },
53560     
53561     onRegionCollapsed : function(region){
53562         this.state[region.getPosition()].collapsed = true;
53563         this.storeState();
53564     },
53565     
53566     onRegionExpanded : function(region){
53567         this.state[region.getPosition()].collapsed = false;
53568         this.storeState();
53569     }
53570 };/*
53571  * Based on:
53572  * Ext JS Library 1.1.1
53573  * Copyright(c) 2006-2007, Ext JS, LLC.
53574  *
53575  * Originally Released Under LGPL - original licence link has changed is not relivant.
53576  *
53577  * Fork - LGPL
53578  * <script type="text/javascript">
53579  */
53580 /**
53581  * @class Roo.ContentPanel
53582  * @extends Roo.util.Observable
53583  * A basic ContentPanel element.
53584  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53585  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53586  * @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
53587  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53588  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53589  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53590  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53591  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53592  * @cfg {String} title          The title for this panel
53593  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53594  * @cfg {String} url            Calls {@link #setUrl} with this value
53595  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
53596  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
53597  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
53598  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
53599
53600  * @constructor
53601  * Create a new ContentPanel.
53602  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
53603  * @param {String/Object} config A string to set only the title or a config object
53604  * @param {String} content (optional) Set the HTML content for this panel
53605  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
53606  */
53607 Roo.ContentPanel = function(el, config, content){
53608     
53609      
53610     /*
53611     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
53612         config = el;
53613         el = Roo.id();
53614     }
53615     if (config && config.parentLayout) { 
53616         el = config.parentLayout.el.createChild(); 
53617     }
53618     */
53619     if(el.autoCreate){ // xtype is available if this is called from factory
53620         config = el;
53621         el = Roo.id();
53622     }
53623     this.el = Roo.get(el);
53624     if(!this.el && config && config.autoCreate){
53625         if(typeof config.autoCreate == "object"){
53626             if(!config.autoCreate.id){
53627                 config.autoCreate.id = config.id||el;
53628             }
53629             this.el = Roo.DomHelper.append(document.body,
53630                         config.autoCreate, true);
53631         }else{
53632             this.el = Roo.DomHelper.append(document.body,
53633                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
53634         }
53635     }
53636     this.closable = false;
53637     this.loaded = false;
53638     this.active = false;
53639     if(typeof config == "string"){
53640         this.title = config;
53641     }else{
53642         Roo.apply(this, config);
53643     }
53644     
53645     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
53646         this.wrapEl = this.el.wrap();
53647         this.toolbar.container = this.el.insertSibling(false, 'before');
53648         this.toolbar = new Roo.Toolbar(this.toolbar);
53649     }
53650     
53651     // xtype created footer. - not sure if will work as we normally have to render first..
53652     if (this.footer && !this.footer.el && this.footer.xtype) {
53653         if (!this.wrapEl) {
53654             this.wrapEl = this.el.wrap();
53655         }
53656     
53657         this.footer.container = this.wrapEl.createChild();
53658          
53659         this.footer = Roo.factory(this.footer, Roo);
53660         
53661     }
53662     
53663     if(this.resizeEl){
53664         this.resizeEl = Roo.get(this.resizeEl, true);
53665     }else{
53666         this.resizeEl = this.el;
53667     }
53668     // handle view.xtype
53669     
53670  
53671     
53672     
53673     this.addEvents({
53674         /**
53675          * @event activate
53676          * Fires when this panel is activated. 
53677          * @param {Roo.ContentPanel} this
53678          */
53679         "activate" : true,
53680         /**
53681          * @event deactivate
53682          * Fires when this panel is activated. 
53683          * @param {Roo.ContentPanel} this
53684          */
53685         "deactivate" : true,
53686
53687         /**
53688          * @event resize
53689          * Fires when this panel is resized if fitToFrame is true.
53690          * @param {Roo.ContentPanel} this
53691          * @param {Number} width The width after any component adjustments
53692          * @param {Number} height The height after any component adjustments
53693          */
53694         "resize" : true,
53695         
53696          /**
53697          * @event render
53698          * Fires when this tab is created
53699          * @param {Roo.ContentPanel} this
53700          */
53701         "render" : true
53702          
53703         
53704     });
53705     
53706
53707     
53708     
53709     if(this.autoScroll){
53710         this.resizeEl.setStyle("overflow", "auto");
53711     } else {
53712         // fix randome scrolling
53713         this.el.on('scroll', function() {
53714             Roo.log('fix random scolling');
53715             this.scrollTo('top',0); 
53716         });
53717     }
53718     content = content || this.content;
53719     if(content){
53720         this.setContent(content);
53721     }
53722     if(config && config.url){
53723         this.setUrl(this.url, this.params, this.loadOnce);
53724     }
53725     
53726     
53727     
53728     Roo.ContentPanel.superclass.constructor.call(this);
53729     
53730     if (this.view && typeof(this.view.xtype) != 'undefined') {
53731         this.view.el = this.el.appendChild(document.createElement("div"));
53732         this.view = Roo.factory(this.view); 
53733         this.view.render  &&  this.view.render(false, '');  
53734     }
53735     
53736     
53737     this.fireEvent('render', this);
53738 };
53739
53740 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
53741     tabTip:'',
53742     setRegion : function(region){
53743         this.region = region;
53744         if(region){
53745            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
53746         }else{
53747            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
53748         } 
53749     },
53750     
53751     /**
53752      * Returns the toolbar for this Panel if one was configured. 
53753      * @return {Roo.Toolbar} 
53754      */
53755     getToolbar : function(){
53756         return this.toolbar;
53757     },
53758     
53759     setActiveState : function(active){
53760         this.active = active;
53761         if(!active){
53762             this.fireEvent("deactivate", this);
53763         }else{
53764             this.fireEvent("activate", this);
53765         }
53766     },
53767     /**
53768      * Updates this panel's element
53769      * @param {String} content The new content
53770      * @param {Boolean} loadScripts (optional) true to look for and process scripts
53771     */
53772     setContent : function(content, loadScripts){
53773         this.el.update(content, loadScripts);
53774     },
53775
53776     ignoreResize : function(w, h){
53777         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
53778             return true;
53779         }else{
53780             this.lastSize = {width: w, height: h};
53781             return false;
53782         }
53783     },
53784     /**
53785      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
53786      * @return {Roo.UpdateManager} The UpdateManager
53787      */
53788     getUpdateManager : function(){
53789         return this.el.getUpdateManager();
53790     },
53791      /**
53792      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
53793      * @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:
53794 <pre><code>
53795 panel.load({
53796     url: "your-url.php",
53797     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
53798     callback: yourFunction,
53799     scope: yourObject, //(optional scope)
53800     discardUrl: false,
53801     nocache: false,
53802     text: "Loading...",
53803     timeout: 30,
53804     scripts: false
53805 });
53806 </code></pre>
53807      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
53808      * 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.
53809      * @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}
53810      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
53811      * @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.
53812      * @return {Roo.ContentPanel} this
53813      */
53814     load : function(){
53815         var um = this.el.getUpdateManager();
53816         um.update.apply(um, arguments);
53817         return this;
53818     },
53819
53820
53821     /**
53822      * 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.
53823      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
53824      * @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)
53825      * @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)
53826      * @return {Roo.UpdateManager} The UpdateManager
53827      */
53828     setUrl : function(url, params, loadOnce){
53829         if(this.refreshDelegate){
53830             this.removeListener("activate", this.refreshDelegate);
53831         }
53832         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
53833         this.on("activate", this.refreshDelegate);
53834         return this.el.getUpdateManager();
53835     },
53836     
53837     _handleRefresh : function(url, params, loadOnce){
53838         if(!loadOnce || !this.loaded){
53839             var updater = this.el.getUpdateManager();
53840             updater.update(url, params, this._setLoaded.createDelegate(this));
53841         }
53842     },
53843     
53844     _setLoaded : function(){
53845         this.loaded = true;
53846     }, 
53847     
53848     /**
53849      * Returns this panel's id
53850      * @return {String} 
53851      */
53852     getId : function(){
53853         return this.el.id;
53854     },
53855     
53856     /** 
53857      * Returns this panel's element - used by regiosn to add.
53858      * @return {Roo.Element} 
53859      */
53860     getEl : function(){
53861         return this.wrapEl || this.el;
53862     },
53863     
53864     adjustForComponents : function(width, height)
53865     {
53866         //Roo.log('adjustForComponents ');
53867         if(this.resizeEl != this.el){
53868             width -= this.el.getFrameWidth('lr');
53869             height -= this.el.getFrameWidth('tb');
53870         }
53871         if(this.toolbar){
53872             var te = this.toolbar.getEl();
53873             height -= te.getHeight();
53874             te.setWidth(width);
53875         }
53876         if(this.footer){
53877             var te = this.footer.getEl();
53878             //Roo.log("footer:" + te.getHeight());
53879             
53880             height -= te.getHeight();
53881             te.setWidth(width);
53882         }
53883         
53884         
53885         if(this.adjustments){
53886             width += this.adjustments[0];
53887             height += this.adjustments[1];
53888         }
53889         return {"width": width, "height": height};
53890     },
53891     
53892     setSize : function(width, height){
53893         if(this.fitToFrame && !this.ignoreResize(width, height)){
53894             if(this.fitContainer && this.resizeEl != this.el){
53895                 this.el.setSize(width, height);
53896             }
53897             var size = this.adjustForComponents(width, height);
53898             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
53899             this.fireEvent('resize', this, size.width, size.height);
53900         }
53901     },
53902     
53903     /**
53904      * Returns this panel's title
53905      * @return {String} 
53906      */
53907     getTitle : function(){
53908         return this.title;
53909     },
53910     
53911     /**
53912      * Set this panel's title
53913      * @param {String} title
53914      */
53915     setTitle : function(title){
53916         this.title = title;
53917         if(this.region){
53918             this.region.updatePanelTitle(this, title);
53919         }
53920     },
53921     
53922     /**
53923      * Returns true is this panel was configured to be closable
53924      * @return {Boolean} 
53925      */
53926     isClosable : function(){
53927         return this.closable;
53928     },
53929     
53930     beforeSlide : function(){
53931         this.el.clip();
53932         this.resizeEl.clip();
53933     },
53934     
53935     afterSlide : function(){
53936         this.el.unclip();
53937         this.resizeEl.unclip();
53938     },
53939     
53940     /**
53941      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
53942      *   Will fail silently if the {@link #setUrl} method has not been called.
53943      *   This does not activate the panel, just updates its content.
53944      */
53945     refresh : function(){
53946         if(this.refreshDelegate){
53947            this.loaded = false;
53948            this.refreshDelegate();
53949         }
53950     },
53951     
53952     /**
53953      * Destroys this panel
53954      */
53955     destroy : function(){
53956         this.el.removeAllListeners();
53957         var tempEl = document.createElement("span");
53958         tempEl.appendChild(this.el.dom);
53959         tempEl.innerHTML = "";
53960         this.el.remove();
53961         this.el = null;
53962     },
53963     
53964     /**
53965      * form - if the content panel contains a form - this is a reference to it.
53966      * @type {Roo.form.Form}
53967      */
53968     form : false,
53969     /**
53970      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
53971      *    This contains a reference to it.
53972      * @type {Roo.View}
53973      */
53974     view : false,
53975     
53976       /**
53977      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
53978      * <pre><code>
53979
53980 layout.addxtype({
53981        xtype : 'Form',
53982        items: [ .... ]
53983    }
53984 );
53985
53986 </code></pre>
53987      * @param {Object} cfg Xtype definition of item to add.
53988      */
53989     
53990     addxtype : function(cfg) {
53991         // add form..
53992         if (cfg.xtype.match(/^Form$/)) {
53993             
53994             var el;
53995             //if (this.footer) {
53996             //    el = this.footer.container.insertSibling(false, 'before');
53997             //} else {
53998                 el = this.el.createChild();
53999             //}
54000
54001             this.form = new  Roo.form.Form(cfg);
54002             
54003             
54004             if ( this.form.allItems.length) {
54005                 this.form.render(el.dom);
54006             }
54007             return this.form;
54008         }
54009         // should only have one of theses..
54010         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54011             // views.. should not be just added - used named prop 'view''
54012             
54013             cfg.el = this.el.appendChild(document.createElement("div"));
54014             // factory?
54015             
54016             var ret = new Roo.factory(cfg);
54017              
54018              ret.render && ret.render(false, ''); // render blank..
54019             this.view = ret;
54020             return ret;
54021         }
54022         return false;
54023     }
54024 });
54025
54026 /**
54027  * @class Roo.GridPanel
54028  * @extends Roo.ContentPanel
54029  * @constructor
54030  * Create a new GridPanel.
54031  * @param {Roo.grid.Grid} grid The grid for this panel
54032  * @param {String/Object} config A string to set only the panel's title, or a config object
54033  */
54034 Roo.GridPanel = function(grid, config){
54035     
54036   
54037     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54038         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54039         
54040     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54041     
54042     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54043     
54044     if(this.toolbar){
54045         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54046     }
54047     // xtype created footer. - not sure if will work as we normally have to render first..
54048     if (this.footer && !this.footer.el && this.footer.xtype) {
54049         
54050         this.footer.container = this.grid.getView().getFooterPanel(true);
54051         this.footer.dataSource = this.grid.dataSource;
54052         this.footer = Roo.factory(this.footer, Roo);
54053         
54054     }
54055     
54056     grid.monitorWindowResize = false; // turn off autosizing
54057     grid.autoHeight = false;
54058     grid.autoWidth = false;
54059     this.grid = grid;
54060     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54061 };
54062
54063 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54064     getId : function(){
54065         return this.grid.id;
54066     },
54067     
54068     /**
54069      * Returns the grid for this panel
54070      * @return {Roo.grid.Grid} 
54071      */
54072     getGrid : function(){
54073         return this.grid;    
54074     },
54075     
54076     setSize : function(width, height){
54077         if(!this.ignoreResize(width, height)){
54078             var grid = this.grid;
54079             var size = this.adjustForComponents(width, height);
54080             grid.getGridEl().setSize(size.width, size.height);
54081             grid.autoSize();
54082         }
54083     },
54084     
54085     beforeSlide : function(){
54086         this.grid.getView().scroller.clip();
54087     },
54088     
54089     afterSlide : function(){
54090         this.grid.getView().scroller.unclip();
54091     },
54092     
54093     destroy : function(){
54094         this.grid.destroy();
54095         delete this.grid;
54096         Roo.GridPanel.superclass.destroy.call(this); 
54097     }
54098 });
54099
54100
54101 /**
54102  * @class Roo.NestedLayoutPanel
54103  * @extends Roo.ContentPanel
54104  * @constructor
54105  * Create a new NestedLayoutPanel.
54106  * 
54107  * 
54108  * @param {Roo.BorderLayout} layout The layout for this panel
54109  * @param {String/Object} config A string to set only the title or a config object
54110  */
54111 Roo.NestedLayoutPanel = function(layout, config)
54112 {
54113     // construct with only one argument..
54114     /* FIXME - implement nicer consturctors
54115     if (layout.layout) {
54116         config = layout;
54117         layout = config.layout;
54118         delete config.layout;
54119     }
54120     if (layout.xtype && !layout.getEl) {
54121         // then layout needs constructing..
54122         layout = Roo.factory(layout, Roo);
54123     }
54124     */
54125     
54126     
54127     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54128     
54129     layout.monitorWindowResize = false; // turn off autosizing
54130     this.layout = layout;
54131     this.layout.getEl().addClass("x-layout-nested-layout");
54132     
54133     
54134     
54135     
54136 };
54137
54138 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54139
54140     setSize : function(width, height){
54141         if(!this.ignoreResize(width, height)){
54142             var size = this.adjustForComponents(width, height);
54143             var el = this.layout.getEl();
54144             el.setSize(size.width, size.height);
54145             var touch = el.dom.offsetWidth;
54146             this.layout.layout();
54147             // ie requires a double layout on the first pass
54148             if(Roo.isIE && !this.initialized){
54149                 this.initialized = true;
54150                 this.layout.layout();
54151             }
54152         }
54153     },
54154     
54155     // activate all subpanels if not currently active..
54156     
54157     setActiveState : function(active){
54158         this.active = active;
54159         if(!active){
54160             this.fireEvent("deactivate", this);
54161             return;
54162         }
54163         
54164         this.fireEvent("activate", this);
54165         // not sure if this should happen before or after..
54166         if (!this.layout) {
54167             return; // should not happen..
54168         }
54169         var reg = false;
54170         for (var r in this.layout.regions) {
54171             reg = this.layout.getRegion(r);
54172             if (reg.getActivePanel()) {
54173                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54174                 reg.setActivePanel(reg.getActivePanel());
54175                 continue;
54176             }
54177             if (!reg.panels.length) {
54178                 continue;
54179             }
54180             reg.showPanel(reg.getPanel(0));
54181         }
54182         
54183         
54184         
54185         
54186     },
54187     
54188     /**
54189      * Returns the nested BorderLayout for this panel
54190      * @return {Roo.BorderLayout} 
54191      */
54192     getLayout : function(){
54193         return this.layout;
54194     },
54195     
54196      /**
54197      * Adds a xtype elements to the layout of the nested panel
54198      * <pre><code>
54199
54200 panel.addxtype({
54201        xtype : 'ContentPanel',
54202        region: 'west',
54203        items: [ .... ]
54204    }
54205 );
54206
54207 panel.addxtype({
54208         xtype : 'NestedLayoutPanel',
54209         region: 'west',
54210         layout: {
54211            center: { },
54212            west: { }   
54213         },
54214         items : [ ... list of content panels or nested layout panels.. ]
54215    }
54216 );
54217 </code></pre>
54218      * @param {Object} cfg Xtype definition of item to add.
54219      */
54220     addxtype : function(cfg) {
54221         return this.layout.addxtype(cfg);
54222     
54223     }
54224 });
54225
54226 Roo.ScrollPanel = function(el, config, content){
54227     config = config || {};
54228     config.fitToFrame = true;
54229     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54230     
54231     this.el.dom.style.overflow = "hidden";
54232     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54233     this.el.removeClass("x-layout-inactive-content");
54234     this.el.on("mousewheel", this.onWheel, this);
54235
54236     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54237     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54238     up.unselectable(); down.unselectable();
54239     up.on("click", this.scrollUp, this);
54240     down.on("click", this.scrollDown, this);
54241     up.addClassOnOver("x-scroller-btn-over");
54242     down.addClassOnOver("x-scroller-btn-over");
54243     up.addClassOnClick("x-scroller-btn-click");
54244     down.addClassOnClick("x-scroller-btn-click");
54245     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54246
54247     this.resizeEl = this.el;
54248     this.el = wrap; this.up = up; this.down = down;
54249 };
54250
54251 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54252     increment : 100,
54253     wheelIncrement : 5,
54254     scrollUp : function(){
54255         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54256     },
54257
54258     scrollDown : function(){
54259         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54260     },
54261
54262     afterScroll : function(){
54263         var el = this.resizeEl;
54264         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54265         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54266         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54267     },
54268
54269     setSize : function(){
54270         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54271         this.afterScroll();
54272     },
54273
54274     onWheel : function(e){
54275         var d = e.getWheelDelta();
54276         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54277         this.afterScroll();
54278         e.stopEvent();
54279     },
54280
54281     setContent : function(content, loadScripts){
54282         this.resizeEl.update(content, loadScripts);
54283     }
54284
54285 });
54286
54287
54288
54289
54290
54291
54292
54293
54294
54295 /**
54296  * @class Roo.TreePanel
54297  * @extends Roo.ContentPanel
54298  * @constructor
54299  * Create a new TreePanel. - defaults to fit/scoll contents.
54300  * @param {String/Object} config A string to set only the panel's title, or a config object
54301  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54302  */
54303 Roo.TreePanel = function(config){
54304     var el = config.el;
54305     var tree = config.tree;
54306     delete config.tree; 
54307     delete config.el; // hopefull!
54308     
54309     // wrapper for IE7 strict & safari scroll issue
54310     
54311     var treeEl = el.createChild();
54312     config.resizeEl = treeEl;
54313     
54314     
54315     
54316     Roo.TreePanel.superclass.constructor.call(this, el, config);
54317  
54318  
54319     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54320     //console.log(tree);
54321     this.on('activate', function()
54322     {
54323         if (this.tree.rendered) {
54324             return;
54325         }
54326         //console.log('render tree');
54327         this.tree.render();
54328     });
54329     // this should not be needed.. - it's actually the 'el' that resizes?
54330     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54331     
54332     //this.on('resize',  function (cp, w, h) {
54333     //        this.tree.innerCt.setWidth(w);
54334     //        this.tree.innerCt.setHeight(h);
54335     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54336     //});
54337
54338         
54339     
54340 };
54341
54342 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54343     fitToFrame : true,
54344     autoScroll : true
54345 });
54346
54347
54348
54349
54350
54351
54352
54353
54354
54355
54356
54357 /*
54358  * Based on:
54359  * Ext JS Library 1.1.1
54360  * Copyright(c) 2006-2007, Ext JS, LLC.
54361  *
54362  * Originally Released Under LGPL - original licence link has changed is not relivant.
54363  *
54364  * Fork - LGPL
54365  * <script type="text/javascript">
54366  */
54367  
54368
54369 /**
54370  * @class Roo.ReaderLayout
54371  * @extends Roo.BorderLayout
54372  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54373  * center region containing two nested regions (a top one for a list view and one for item preview below),
54374  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54375  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54376  * expedites the setup of the overall layout and regions for this common application style.
54377  * Example:
54378  <pre><code>
54379 var reader = new Roo.ReaderLayout();
54380 var CP = Roo.ContentPanel;  // shortcut for adding
54381
54382 reader.beginUpdate();
54383 reader.add("north", new CP("north", "North"));
54384 reader.add("west", new CP("west", {title: "West"}));
54385 reader.add("east", new CP("east", {title: "East"}));
54386
54387 reader.regions.listView.add(new CP("listView", "List"));
54388 reader.regions.preview.add(new CP("preview", "Preview"));
54389 reader.endUpdate();
54390 </code></pre>
54391 * @constructor
54392 * Create a new ReaderLayout
54393 * @param {Object} config Configuration options
54394 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54395 * document.body if omitted)
54396 */
54397 Roo.ReaderLayout = function(config, renderTo){
54398     var c = config || {size:{}};
54399     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54400         north: c.north !== false ? Roo.apply({
54401             split:false,
54402             initialSize: 32,
54403             titlebar: false
54404         }, c.north) : false,
54405         west: c.west !== false ? Roo.apply({
54406             split:true,
54407             initialSize: 200,
54408             minSize: 175,
54409             maxSize: 400,
54410             titlebar: true,
54411             collapsible: true,
54412             animate: true,
54413             margins:{left:5,right:0,bottom:5,top:5},
54414             cmargins:{left:5,right:5,bottom:5,top:5}
54415         }, c.west) : false,
54416         east: c.east !== false ? Roo.apply({
54417             split:true,
54418             initialSize: 200,
54419             minSize: 175,
54420             maxSize: 400,
54421             titlebar: true,
54422             collapsible: true,
54423             animate: true,
54424             margins:{left:0,right:5,bottom:5,top:5},
54425             cmargins:{left:5,right:5,bottom:5,top:5}
54426         }, c.east) : false,
54427         center: Roo.apply({
54428             tabPosition: 'top',
54429             autoScroll:false,
54430             closeOnTab: true,
54431             titlebar:false,
54432             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54433         }, c.center)
54434     });
54435
54436     this.el.addClass('x-reader');
54437
54438     this.beginUpdate();
54439
54440     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54441         south: c.preview !== false ? Roo.apply({
54442             split:true,
54443             initialSize: 200,
54444             minSize: 100,
54445             autoScroll:true,
54446             collapsible:true,
54447             titlebar: true,
54448             cmargins:{top:5,left:0, right:0, bottom:0}
54449         }, c.preview) : false,
54450         center: Roo.apply({
54451             autoScroll:false,
54452             titlebar:false,
54453             minHeight:200
54454         }, c.listView)
54455     });
54456     this.add('center', new Roo.NestedLayoutPanel(inner,
54457             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54458
54459     this.endUpdate();
54460
54461     this.regions.preview = inner.getRegion('south');
54462     this.regions.listView = inner.getRegion('center');
54463 };
54464
54465 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54466  * Based on:
54467  * Ext JS Library 1.1.1
54468  * Copyright(c) 2006-2007, Ext JS, LLC.
54469  *
54470  * Originally Released Under LGPL - original licence link has changed is not relivant.
54471  *
54472  * Fork - LGPL
54473  * <script type="text/javascript">
54474  */
54475  
54476 /**
54477  * @class Roo.grid.Grid
54478  * @extends Roo.util.Observable
54479  * This class represents the primary interface of a component based grid control.
54480  * <br><br>Usage:<pre><code>
54481  var grid = new Roo.grid.Grid("my-container-id", {
54482      ds: myDataStore,
54483      cm: myColModel,
54484      selModel: mySelectionModel,
54485      autoSizeColumns: true,
54486      monitorWindowResize: false,
54487      trackMouseOver: true
54488  });
54489  // set any options
54490  grid.render();
54491  * </code></pre>
54492  * <b>Common Problems:</b><br/>
54493  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54494  * element will correct this<br/>
54495  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54496  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54497  * are unpredictable.<br/>
54498  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54499  * grid to calculate dimensions/offsets.<br/>
54500   * @constructor
54501  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54502  * The container MUST have some type of size defined for the grid to fill. The container will be
54503  * automatically set to position relative if it isn't already.
54504  * @param {Object} config A config object that sets properties on this grid.
54505  */
54506 Roo.grid.Grid = function(container, config){
54507         // initialize the container
54508         this.container = Roo.get(container);
54509         this.container.update("");
54510         this.container.setStyle("overflow", "hidden");
54511     this.container.addClass('x-grid-container');
54512
54513     this.id = this.container.id;
54514
54515     Roo.apply(this, config);
54516     // check and correct shorthanded configs
54517     if(this.ds){
54518         this.dataSource = this.ds;
54519         delete this.ds;
54520     }
54521     if(this.cm){
54522         this.colModel = this.cm;
54523         delete this.cm;
54524     }
54525     if(this.sm){
54526         this.selModel = this.sm;
54527         delete this.sm;
54528     }
54529
54530     if (this.selModel) {
54531         this.selModel = Roo.factory(this.selModel, Roo.grid);
54532         this.sm = this.selModel;
54533         this.sm.xmodule = this.xmodule || false;
54534     }
54535     if (typeof(this.colModel.config) == 'undefined') {
54536         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54537         this.cm = this.colModel;
54538         this.cm.xmodule = this.xmodule || false;
54539     }
54540     if (this.dataSource) {
54541         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54542         this.ds = this.dataSource;
54543         this.ds.xmodule = this.xmodule || false;
54544          
54545     }
54546     
54547     
54548     
54549     if(this.width){
54550         this.container.setWidth(this.width);
54551     }
54552
54553     if(this.height){
54554         this.container.setHeight(this.height);
54555     }
54556     /** @private */
54557         this.addEvents({
54558         // raw events
54559         /**
54560          * @event click
54561          * The raw click event for the entire grid.
54562          * @param {Roo.EventObject} e
54563          */
54564         "click" : true,
54565         /**
54566          * @event dblclick
54567          * The raw dblclick event for the entire grid.
54568          * @param {Roo.EventObject} e
54569          */
54570         "dblclick" : true,
54571         /**
54572          * @event contextmenu
54573          * The raw contextmenu event for the entire grid.
54574          * @param {Roo.EventObject} e
54575          */
54576         "contextmenu" : true,
54577         /**
54578          * @event mousedown
54579          * The raw mousedown event for the entire grid.
54580          * @param {Roo.EventObject} e
54581          */
54582         "mousedown" : true,
54583         /**
54584          * @event mouseup
54585          * The raw mouseup event for the entire grid.
54586          * @param {Roo.EventObject} e
54587          */
54588         "mouseup" : true,
54589         /**
54590          * @event mouseover
54591          * The raw mouseover event for the entire grid.
54592          * @param {Roo.EventObject} e
54593          */
54594         "mouseover" : true,
54595         /**
54596          * @event mouseout
54597          * The raw mouseout event for the entire grid.
54598          * @param {Roo.EventObject} e
54599          */
54600         "mouseout" : true,
54601         /**
54602          * @event keypress
54603          * The raw keypress event for the entire grid.
54604          * @param {Roo.EventObject} e
54605          */
54606         "keypress" : true,
54607         /**
54608          * @event keydown
54609          * The raw keydown event for the entire grid.
54610          * @param {Roo.EventObject} e
54611          */
54612         "keydown" : true,
54613
54614         // custom events
54615
54616         /**
54617          * @event cellclick
54618          * Fires when a cell is clicked
54619          * @param {Grid} this
54620          * @param {Number} rowIndex
54621          * @param {Number} columnIndex
54622          * @param {Roo.EventObject} e
54623          */
54624         "cellclick" : true,
54625         /**
54626          * @event celldblclick
54627          * Fires when a cell is double clicked
54628          * @param {Grid} this
54629          * @param {Number} rowIndex
54630          * @param {Number} columnIndex
54631          * @param {Roo.EventObject} e
54632          */
54633         "celldblclick" : true,
54634         /**
54635          * @event rowclick
54636          * Fires when a row is clicked
54637          * @param {Grid} this
54638          * @param {Number} rowIndex
54639          * @param {Roo.EventObject} e
54640          */
54641         "rowclick" : true,
54642         /**
54643          * @event rowdblclick
54644          * Fires when a row is double clicked
54645          * @param {Grid} this
54646          * @param {Number} rowIndex
54647          * @param {Roo.EventObject} e
54648          */
54649         "rowdblclick" : true,
54650         /**
54651          * @event headerclick
54652          * Fires when a header is clicked
54653          * @param {Grid} this
54654          * @param {Number} columnIndex
54655          * @param {Roo.EventObject} e
54656          */
54657         "headerclick" : true,
54658         /**
54659          * @event headerdblclick
54660          * Fires when a header cell is double clicked
54661          * @param {Grid} this
54662          * @param {Number} columnIndex
54663          * @param {Roo.EventObject} e
54664          */
54665         "headerdblclick" : true,
54666         /**
54667          * @event rowcontextmenu
54668          * Fires when a row is right clicked
54669          * @param {Grid} this
54670          * @param {Number} rowIndex
54671          * @param {Roo.EventObject} e
54672          */
54673         "rowcontextmenu" : true,
54674         /**
54675          * @event cellcontextmenu
54676          * Fires when a cell is right clicked
54677          * @param {Grid} this
54678          * @param {Number} rowIndex
54679          * @param {Number} cellIndex
54680          * @param {Roo.EventObject} e
54681          */
54682          "cellcontextmenu" : true,
54683         /**
54684          * @event headercontextmenu
54685          * Fires when a header is right clicked
54686          * @param {Grid} this
54687          * @param {Number} columnIndex
54688          * @param {Roo.EventObject} e
54689          */
54690         "headercontextmenu" : true,
54691         /**
54692          * @event bodyscroll
54693          * Fires when the body element is scrolled
54694          * @param {Number} scrollLeft
54695          * @param {Number} scrollTop
54696          */
54697         "bodyscroll" : true,
54698         /**
54699          * @event columnresize
54700          * Fires when the user resizes a column
54701          * @param {Number} columnIndex
54702          * @param {Number} newSize
54703          */
54704         "columnresize" : true,
54705         /**
54706          * @event columnmove
54707          * Fires when the user moves a column
54708          * @param {Number} oldIndex
54709          * @param {Number} newIndex
54710          */
54711         "columnmove" : true,
54712         /**
54713          * @event startdrag
54714          * Fires when row(s) start being dragged
54715          * @param {Grid} this
54716          * @param {Roo.GridDD} dd The drag drop object
54717          * @param {event} e The raw browser event
54718          */
54719         "startdrag" : true,
54720         /**
54721          * @event enddrag
54722          * Fires when a drag operation is complete
54723          * @param {Grid} this
54724          * @param {Roo.GridDD} dd The drag drop object
54725          * @param {event} e The raw browser event
54726          */
54727         "enddrag" : true,
54728         /**
54729          * @event dragdrop
54730          * Fires when dragged row(s) are dropped on a valid DD target
54731          * @param {Grid} this
54732          * @param {Roo.GridDD} dd The drag drop object
54733          * @param {String} targetId The target drag drop object
54734          * @param {event} e The raw browser event
54735          */
54736         "dragdrop" : true,
54737         /**
54738          * @event dragover
54739          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
54740          * @param {Grid} this
54741          * @param {Roo.GridDD} dd The drag drop object
54742          * @param {String} targetId The target drag drop object
54743          * @param {event} e The raw browser event
54744          */
54745         "dragover" : true,
54746         /**
54747          * @event dragenter
54748          *  Fires when the dragged row(s) first cross another DD target while being dragged
54749          * @param {Grid} this
54750          * @param {Roo.GridDD} dd The drag drop object
54751          * @param {String} targetId The target drag drop object
54752          * @param {event} e The raw browser event
54753          */
54754         "dragenter" : true,
54755         /**
54756          * @event dragout
54757          * Fires when the dragged row(s) leave another DD target while being dragged
54758          * @param {Grid} this
54759          * @param {Roo.GridDD} dd The drag drop object
54760          * @param {String} targetId The target drag drop object
54761          * @param {event} e The raw browser event
54762          */
54763         "dragout" : true,
54764         /**
54765          * @event rowclass
54766          * Fires when a row is rendered, so you can change add a style to it.
54767          * @param {GridView} gridview   The grid view
54768          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
54769          */
54770         'rowclass' : true,
54771
54772         /**
54773          * @event render
54774          * Fires when the grid is rendered
54775          * @param {Grid} grid
54776          */
54777         'render' : true
54778     });
54779
54780     Roo.grid.Grid.superclass.constructor.call(this);
54781 };
54782 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
54783     
54784     /**
54785      * @cfg {String} ddGroup - drag drop group.
54786      */
54787
54788     /**
54789      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
54790      */
54791     minColumnWidth : 25,
54792
54793     /**
54794      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
54795      * <b>on initial render.</b> It is more efficient to explicitly size the columns
54796      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
54797      */
54798     autoSizeColumns : false,
54799
54800     /**
54801      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
54802      */
54803     autoSizeHeaders : true,
54804
54805     /**
54806      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
54807      */
54808     monitorWindowResize : true,
54809
54810     /**
54811      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
54812      * rows measured to get a columns size. Default is 0 (all rows).
54813      */
54814     maxRowsToMeasure : 0,
54815
54816     /**
54817      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
54818      */
54819     trackMouseOver : true,
54820
54821     /**
54822     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
54823     */
54824     
54825     /**
54826     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
54827     */
54828     enableDragDrop : false,
54829     
54830     /**
54831     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
54832     */
54833     enableColumnMove : true,
54834     
54835     /**
54836     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
54837     */
54838     enableColumnHide : true,
54839     
54840     /**
54841     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
54842     */
54843     enableRowHeightSync : false,
54844     
54845     /**
54846     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
54847     */
54848     stripeRows : true,
54849     
54850     /**
54851     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
54852     */
54853     autoHeight : false,
54854
54855     /**
54856      * @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.
54857      */
54858     autoExpandColumn : false,
54859
54860     /**
54861     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
54862     * Default is 50.
54863     */
54864     autoExpandMin : 50,
54865
54866     /**
54867     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
54868     */
54869     autoExpandMax : 1000,
54870
54871     /**
54872     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
54873     */
54874     view : null,
54875
54876     /**
54877     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
54878     */
54879     loadMask : false,
54880     /**
54881     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
54882     */
54883     dropTarget: false,
54884     
54885    
54886     
54887     // private
54888     rendered : false,
54889
54890     /**
54891     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
54892     * of a fixed width. Default is false.
54893     */
54894     /**
54895     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
54896     */
54897     /**
54898      * Called once after all setup has been completed and the grid is ready to be rendered.
54899      * @return {Roo.grid.Grid} this
54900      */
54901     render : function()
54902     {
54903         var c = this.container;
54904         // try to detect autoHeight/width mode
54905         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
54906             this.autoHeight = true;
54907         }
54908         var view = this.getView();
54909         view.init(this);
54910
54911         c.on("click", this.onClick, this);
54912         c.on("dblclick", this.onDblClick, this);
54913         c.on("contextmenu", this.onContextMenu, this);
54914         c.on("keydown", this.onKeyDown, this);
54915         if (Roo.isTouch) {
54916             c.on("touchstart", this.onTouchStart, this);
54917         }
54918
54919         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
54920
54921         this.getSelectionModel().init(this);
54922
54923         view.render();
54924
54925         if(this.loadMask){
54926             this.loadMask = new Roo.LoadMask(this.container,
54927                     Roo.apply({store:this.dataSource}, this.loadMask));
54928         }
54929         
54930         
54931         if (this.toolbar && this.toolbar.xtype) {
54932             this.toolbar.container = this.getView().getHeaderPanel(true);
54933             this.toolbar = new Roo.Toolbar(this.toolbar);
54934         }
54935         if (this.footer && this.footer.xtype) {
54936             this.footer.dataSource = this.getDataSource();
54937             this.footer.container = this.getView().getFooterPanel(true);
54938             this.footer = Roo.factory(this.footer, Roo);
54939         }
54940         if (this.dropTarget && this.dropTarget.xtype) {
54941             delete this.dropTarget.xtype;
54942             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
54943         }
54944         
54945         
54946         this.rendered = true;
54947         this.fireEvent('render', this);
54948         return this;
54949     },
54950
54951         /**
54952          * Reconfigures the grid to use a different Store and Column Model.
54953          * The View will be bound to the new objects and refreshed.
54954          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
54955          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
54956          */
54957     reconfigure : function(dataSource, colModel){
54958         if(this.loadMask){
54959             this.loadMask.destroy();
54960             this.loadMask = new Roo.LoadMask(this.container,
54961                     Roo.apply({store:dataSource}, this.loadMask));
54962         }
54963         this.view.bind(dataSource, colModel);
54964         this.dataSource = dataSource;
54965         this.colModel = colModel;
54966         this.view.refresh(true);
54967     },
54968
54969     // private
54970     onKeyDown : function(e){
54971         this.fireEvent("keydown", e);
54972     },
54973
54974     /**
54975      * Destroy this grid.
54976      * @param {Boolean} removeEl True to remove the element
54977      */
54978     destroy : function(removeEl, keepListeners){
54979         if(this.loadMask){
54980             this.loadMask.destroy();
54981         }
54982         var c = this.container;
54983         c.removeAllListeners();
54984         this.view.destroy();
54985         this.colModel.purgeListeners();
54986         if(!keepListeners){
54987             this.purgeListeners();
54988         }
54989         c.update("");
54990         if(removeEl === true){
54991             c.remove();
54992         }
54993     },
54994
54995     // private
54996     processEvent : function(name, e){
54997         // does this fire select???
54998         //Roo.log('grid:processEvent '  + name);
54999         
55000         if (name != 'touchstart' ) {
55001             this.fireEvent(name, e);    
55002         }
55003         
55004         var t = e.getTarget();
55005         var v = this.view;
55006         var header = v.findHeaderIndex(t);
55007         if(header !== false){
55008             var ename = name == 'touchstart' ? 'click' : name;
55009              
55010             this.fireEvent("header" + ename, this, header, e);
55011         }else{
55012             var row = v.findRowIndex(t);
55013             var cell = v.findCellIndex(t);
55014             if (name == 'touchstart') {
55015                 // first touch is always a click.
55016                 // hopefull this happens after selection is updated.?
55017                 name = false;
55018                 
55019                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55020                     var cs = this.selModel.getSelectedCell();
55021                     if (row == cs[0] && cell == cs[1]){
55022                         name = 'dblclick';
55023                     }
55024                 }
55025                 if (typeof(this.selModel.getSelections) != 'undefined') {
55026                     var cs = this.selModel.getSelections();
55027                     var ds = this.dataSource;
55028                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55029                         name = 'dblclick';
55030                     }
55031                 }
55032                 if (!name) {
55033                     return;
55034                 }
55035             }
55036             
55037             
55038             if(row !== false){
55039                 this.fireEvent("row" + name, this, row, e);
55040                 if(cell !== false){
55041                     this.fireEvent("cell" + name, this, row, cell, e);
55042                 }
55043             }
55044         }
55045     },
55046
55047     // private
55048     onClick : function(e){
55049         this.processEvent("click", e);
55050     },
55051    // private
55052     onTouchStart : function(e){
55053         this.processEvent("touchstart", e);
55054     },
55055
55056     // private
55057     onContextMenu : function(e, t){
55058         this.processEvent("contextmenu", e);
55059     },
55060
55061     // private
55062     onDblClick : function(e){
55063         this.processEvent("dblclick", e);
55064     },
55065
55066     // private
55067     walkCells : function(row, col, step, fn, scope){
55068         var cm = this.colModel, clen = cm.getColumnCount();
55069         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55070         if(step < 0){
55071             if(col < 0){
55072                 row--;
55073                 first = false;
55074             }
55075             while(row >= 0){
55076                 if(!first){
55077                     col = clen-1;
55078                 }
55079                 first = false;
55080                 while(col >= 0){
55081                     if(fn.call(scope || this, row, col, cm) === true){
55082                         return [row, col];
55083                     }
55084                     col--;
55085                 }
55086                 row--;
55087             }
55088         } else {
55089             if(col >= clen){
55090                 row++;
55091                 first = false;
55092             }
55093             while(row < rlen){
55094                 if(!first){
55095                     col = 0;
55096                 }
55097                 first = false;
55098                 while(col < clen){
55099                     if(fn.call(scope || this, row, col, cm) === true){
55100                         return [row, col];
55101                     }
55102                     col++;
55103                 }
55104                 row++;
55105             }
55106         }
55107         return null;
55108     },
55109
55110     // private
55111     getSelections : function(){
55112         return this.selModel.getSelections();
55113     },
55114
55115     /**
55116      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55117      * but if manual update is required this method will initiate it.
55118      */
55119     autoSize : function(){
55120         if(this.rendered){
55121             this.view.layout();
55122             if(this.view.adjustForScroll){
55123                 this.view.adjustForScroll();
55124             }
55125         }
55126     },
55127
55128     /**
55129      * Returns the grid's underlying element.
55130      * @return {Element} The element
55131      */
55132     getGridEl : function(){
55133         return this.container;
55134     },
55135
55136     // private for compatibility, overridden by editor grid
55137     stopEditing : function(){},
55138
55139     /**
55140      * Returns the grid's SelectionModel.
55141      * @return {SelectionModel}
55142      */
55143     getSelectionModel : function(){
55144         if(!this.selModel){
55145             this.selModel = new Roo.grid.RowSelectionModel();
55146         }
55147         return this.selModel;
55148     },
55149
55150     /**
55151      * Returns the grid's DataSource.
55152      * @return {DataSource}
55153      */
55154     getDataSource : function(){
55155         return this.dataSource;
55156     },
55157
55158     /**
55159      * Returns the grid's ColumnModel.
55160      * @return {ColumnModel}
55161      */
55162     getColumnModel : function(){
55163         return this.colModel;
55164     },
55165
55166     /**
55167      * Returns the grid's GridView object.
55168      * @return {GridView}
55169      */
55170     getView : function(){
55171         if(!this.view){
55172             this.view = new Roo.grid.GridView(this.viewConfig);
55173         }
55174         return this.view;
55175     },
55176     /**
55177      * Called to get grid's drag proxy text, by default returns this.ddText.
55178      * @return {String}
55179      */
55180     getDragDropText : function(){
55181         var count = this.selModel.getCount();
55182         return String.format(this.ddText, count, count == 1 ? '' : 's');
55183     }
55184 });
55185 /**
55186  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55187  * %0 is replaced with the number of selected rows.
55188  * @type String
55189  */
55190 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55191  * Based on:
55192  * Ext JS Library 1.1.1
55193  * Copyright(c) 2006-2007, Ext JS, LLC.
55194  *
55195  * Originally Released Under LGPL - original licence link has changed is not relivant.
55196  *
55197  * Fork - LGPL
55198  * <script type="text/javascript">
55199  */
55200  
55201 Roo.grid.AbstractGridView = function(){
55202         this.grid = null;
55203         
55204         this.events = {
55205             "beforerowremoved" : true,
55206             "beforerowsinserted" : true,
55207             "beforerefresh" : true,
55208             "rowremoved" : true,
55209             "rowsinserted" : true,
55210             "rowupdated" : true,
55211             "refresh" : true
55212         };
55213     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55214 };
55215
55216 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55217     rowClass : "x-grid-row",
55218     cellClass : "x-grid-cell",
55219     tdClass : "x-grid-td",
55220     hdClass : "x-grid-hd",
55221     splitClass : "x-grid-hd-split",
55222     
55223     init: function(grid){
55224         this.grid = grid;
55225                 var cid = this.grid.getGridEl().id;
55226         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55227         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55228         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55229         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55230         },
55231         
55232     getColumnRenderers : function(){
55233         var renderers = [];
55234         var cm = this.grid.colModel;
55235         var colCount = cm.getColumnCount();
55236         for(var i = 0; i < colCount; i++){
55237             renderers[i] = cm.getRenderer(i);
55238         }
55239         return renderers;
55240     },
55241     
55242     getColumnIds : function(){
55243         var ids = [];
55244         var cm = this.grid.colModel;
55245         var colCount = cm.getColumnCount();
55246         for(var i = 0; i < colCount; i++){
55247             ids[i] = cm.getColumnId(i);
55248         }
55249         return ids;
55250     },
55251     
55252     getDataIndexes : function(){
55253         if(!this.indexMap){
55254             this.indexMap = this.buildIndexMap();
55255         }
55256         return this.indexMap.colToData;
55257     },
55258     
55259     getColumnIndexByDataIndex : function(dataIndex){
55260         if(!this.indexMap){
55261             this.indexMap = this.buildIndexMap();
55262         }
55263         return this.indexMap.dataToCol[dataIndex];
55264     },
55265     
55266     /**
55267      * Set a css style for a column dynamically. 
55268      * @param {Number} colIndex The index of the column
55269      * @param {String} name The css property name
55270      * @param {String} value The css value
55271      */
55272     setCSSStyle : function(colIndex, name, value){
55273         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55274         Roo.util.CSS.updateRule(selector, name, value);
55275     },
55276     
55277     generateRules : function(cm){
55278         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55279         Roo.util.CSS.removeStyleSheet(rulesId);
55280         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55281             var cid = cm.getColumnId(i);
55282             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55283                          this.tdSelector, cid, " {\n}\n",
55284                          this.hdSelector, cid, " {\n}\n",
55285                          this.splitSelector, cid, " {\n}\n");
55286         }
55287         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55288     }
55289 });/*
55290  * Based on:
55291  * Ext JS Library 1.1.1
55292  * Copyright(c) 2006-2007, Ext JS, LLC.
55293  *
55294  * Originally Released Under LGPL - original licence link has changed is not relivant.
55295  *
55296  * Fork - LGPL
55297  * <script type="text/javascript">
55298  */
55299
55300 // private
55301 // This is a support class used internally by the Grid components
55302 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55303     this.grid = grid;
55304     this.view = grid.getView();
55305     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55306     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55307     if(hd2){
55308         this.setHandleElId(Roo.id(hd));
55309         this.setOuterHandleElId(Roo.id(hd2));
55310     }
55311     this.scroll = false;
55312 };
55313 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55314     maxDragWidth: 120,
55315     getDragData : function(e){
55316         var t = Roo.lib.Event.getTarget(e);
55317         var h = this.view.findHeaderCell(t);
55318         if(h){
55319             return {ddel: h.firstChild, header:h};
55320         }
55321         return false;
55322     },
55323
55324     onInitDrag : function(e){
55325         this.view.headersDisabled = true;
55326         var clone = this.dragData.ddel.cloneNode(true);
55327         clone.id = Roo.id();
55328         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55329         this.proxy.update(clone);
55330         return true;
55331     },
55332
55333     afterValidDrop : function(){
55334         var v = this.view;
55335         setTimeout(function(){
55336             v.headersDisabled = false;
55337         }, 50);
55338     },
55339
55340     afterInvalidDrop : function(){
55341         var v = this.view;
55342         setTimeout(function(){
55343             v.headersDisabled = false;
55344         }, 50);
55345     }
55346 });
55347 /*
55348  * Based on:
55349  * Ext JS Library 1.1.1
55350  * Copyright(c) 2006-2007, Ext JS, LLC.
55351  *
55352  * Originally Released Under LGPL - original licence link has changed is not relivant.
55353  *
55354  * Fork - LGPL
55355  * <script type="text/javascript">
55356  */
55357 // private
55358 // This is a support class used internally by the Grid components
55359 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55360     this.grid = grid;
55361     this.view = grid.getView();
55362     // split the proxies so they don't interfere with mouse events
55363     this.proxyTop = Roo.DomHelper.append(document.body, {
55364         cls:"col-move-top", html:"&#160;"
55365     }, true);
55366     this.proxyBottom = Roo.DomHelper.append(document.body, {
55367         cls:"col-move-bottom", html:"&#160;"
55368     }, true);
55369     this.proxyTop.hide = this.proxyBottom.hide = function(){
55370         this.setLeftTop(-100,-100);
55371         this.setStyle("visibility", "hidden");
55372     };
55373     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55374     // temporarily disabled
55375     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55376     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55377 };
55378 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55379     proxyOffsets : [-4, -9],
55380     fly: Roo.Element.fly,
55381
55382     getTargetFromEvent : function(e){
55383         var t = Roo.lib.Event.getTarget(e);
55384         var cindex = this.view.findCellIndex(t);
55385         if(cindex !== false){
55386             return this.view.getHeaderCell(cindex);
55387         }
55388         return null;
55389     },
55390
55391     nextVisible : function(h){
55392         var v = this.view, cm = this.grid.colModel;
55393         h = h.nextSibling;
55394         while(h){
55395             if(!cm.isHidden(v.getCellIndex(h))){
55396                 return h;
55397             }
55398             h = h.nextSibling;
55399         }
55400         return null;
55401     },
55402
55403     prevVisible : function(h){
55404         var v = this.view, cm = this.grid.colModel;
55405         h = h.prevSibling;
55406         while(h){
55407             if(!cm.isHidden(v.getCellIndex(h))){
55408                 return h;
55409             }
55410             h = h.prevSibling;
55411         }
55412         return null;
55413     },
55414
55415     positionIndicator : function(h, n, e){
55416         var x = Roo.lib.Event.getPageX(e);
55417         var r = Roo.lib.Dom.getRegion(n.firstChild);
55418         var px, pt, py = r.top + this.proxyOffsets[1];
55419         if((r.right - x) <= (r.right-r.left)/2){
55420             px = r.right+this.view.borderWidth;
55421             pt = "after";
55422         }else{
55423             px = r.left;
55424             pt = "before";
55425         }
55426         var oldIndex = this.view.getCellIndex(h);
55427         var newIndex = this.view.getCellIndex(n);
55428
55429         if(this.grid.colModel.isFixed(newIndex)){
55430             return false;
55431         }
55432
55433         var locked = this.grid.colModel.isLocked(newIndex);
55434
55435         if(pt == "after"){
55436             newIndex++;
55437         }
55438         if(oldIndex < newIndex){
55439             newIndex--;
55440         }
55441         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55442             return false;
55443         }
55444         px +=  this.proxyOffsets[0];
55445         this.proxyTop.setLeftTop(px, py);
55446         this.proxyTop.show();
55447         if(!this.bottomOffset){
55448             this.bottomOffset = this.view.mainHd.getHeight();
55449         }
55450         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55451         this.proxyBottom.show();
55452         return pt;
55453     },
55454
55455     onNodeEnter : function(n, dd, e, data){
55456         if(data.header != n){
55457             this.positionIndicator(data.header, n, e);
55458         }
55459     },
55460
55461     onNodeOver : function(n, dd, e, data){
55462         var result = false;
55463         if(data.header != n){
55464             result = this.positionIndicator(data.header, n, e);
55465         }
55466         if(!result){
55467             this.proxyTop.hide();
55468             this.proxyBottom.hide();
55469         }
55470         return result ? this.dropAllowed : this.dropNotAllowed;
55471     },
55472
55473     onNodeOut : function(n, dd, e, data){
55474         this.proxyTop.hide();
55475         this.proxyBottom.hide();
55476     },
55477
55478     onNodeDrop : function(n, dd, e, data){
55479         var h = data.header;
55480         if(h != n){
55481             var cm = this.grid.colModel;
55482             var x = Roo.lib.Event.getPageX(e);
55483             var r = Roo.lib.Dom.getRegion(n.firstChild);
55484             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55485             var oldIndex = this.view.getCellIndex(h);
55486             var newIndex = this.view.getCellIndex(n);
55487             var locked = cm.isLocked(newIndex);
55488             if(pt == "after"){
55489                 newIndex++;
55490             }
55491             if(oldIndex < newIndex){
55492                 newIndex--;
55493             }
55494             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55495                 return false;
55496             }
55497             cm.setLocked(oldIndex, locked, true);
55498             cm.moveColumn(oldIndex, newIndex);
55499             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55500             return true;
55501         }
55502         return false;
55503     }
55504 });
55505 /*
55506  * Based on:
55507  * Ext JS Library 1.1.1
55508  * Copyright(c) 2006-2007, Ext JS, LLC.
55509  *
55510  * Originally Released Under LGPL - original licence link has changed is not relivant.
55511  *
55512  * Fork - LGPL
55513  * <script type="text/javascript">
55514  */
55515   
55516 /**
55517  * @class Roo.grid.GridView
55518  * @extends Roo.util.Observable
55519  *
55520  * @constructor
55521  * @param {Object} config
55522  */
55523 Roo.grid.GridView = function(config){
55524     Roo.grid.GridView.superclass.constructor.call(this);
55525     this.el = null;
55526
55527     Roo.apply(this, config);
55528 };
55529
55530 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55531
55532     unselectable :  'unselectable="on"',
55533     unselectableCls :  'x-unselectable',
55534     
55535     
55536     rowClass : "x-grid-row",
55537
55538     cellClass : "x-grid-col",
55539
55540     tdClass : "x-grid-td",
55541
55542     hdClass : "x-grid-hd",
55543
55544     splitClass : "x-grid-split",
55545
55546     sortClasses : ["sort-asc", "sort-desc"],
55547
55548     enableMoveAnim : false,
55549
55550     hlColor: "C3DAF9",
55551
55552     dh : Roo.DomHelper,
55553
55554     fly : Roo.Element.fly,
55555
55556     css : Roo.util.CSS,
55557
55558     borderWidth: 1,
55559
55560     splitOffset: 3,
55561
55562     scrollIncrement : 22,
55563
55564     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55565
55566     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55567
55568     bind : function(ds, cm){
55569         if(this.ds){
55570             this.ds.un("load", this.onLoad, this);
55571             this.ds.un("datachanged", this.onDataChange, this);
55572             this.ds.un("add", this.onAdd, this);
55573             this.ds.un("remove", this.onRemove, this);
55574             this.ds.un("update", this.onUpdate, this);
55575             this.ds.un("clear", this.onClear, this);
55576         }
55577         if(ds){
55578             ds.on("load", this.onLoad, this);
55579             ds.on("datachanged", this.onDataChange, this);
55580             ds.on("add", this.onAdd, this);
55581             ds.on("remove", this.onRemove, this);
55582             ds.on("update", this.onUpdate, this);
55583             ds.on("clear", this.onClear, this);
55584         }
55585         this.ds = ds;
55586
55587         if(this.cm){
55588             this.cm.un("widthchange", this.onColWidthChange, this);
55589             this.cm.un("headerchange", this.onHeaderChange, this);
55590             this.cm.un("hiddenchange", this.onHiddenChange, this);
55591             this.cm.un("columnmoved", this.onColumnMove, this);
55592             this.cm.un("columnlockchange", this.onColumnLock, this);
55593         }
55594         if(cm){
55595             this.generateRules(cm);
55596             cm.on("widthchange", this.onColWidthChange, this);
55597             cm.on("headerchange", this.onHeaderChange, this);
55598             cm.on("hiddenchange", this.onHiddenChange, this);
55599             cm.on("columnmoved", this.onColumnMove, this);
55600             cm.on("columnlockchange", this.onColumnLock, this);
55601         }
55602         this.cm = cm;
55603     },
55604
55605     init: function(grid){
55606         Roo.grid.GridView.superclass.init.call(this, grid);
55607
55608         this.bind(grid.dataSource, grid.colModel);
55609
55610         grid.on("headerclick", this.handleHeaderClick, this);
55611
55612         if(grid.trackMouseOver){
55613             grid.on("mouseover", this.onRowOver, this);
55614             grid.on("mouseout", this.onRowOut, this);
55615         }
55616         grid.cancelTextSelection = function(){};
55617         this.gridId = grid.id;
55618
55619         var tpls = this.templates || {};
55620
55621         if(!tpls.master){
55622             tpls.master = new Roo.Template(
55623                '<div class="x-grid" hidefocus="true">',
55624                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
55625                   '<div class="x-grid-topbar"></div>',
55626                   '<div class="x-grid-scroller"><div></div></div>',
55627                   '<div class="x-grid-locked">',
55628                       '<div class="x-grid-header">{lockedHeader}</div>',
55629                       '<div class="x-grid-body">{lockedBody}</div>',
55630                   "</div>",
55631                   '<div class="x-grid-viewport">',
55632                       '<div class="x-grid-header">{header}</div>',
55633                       '<div class="x-grid-body">{body}</div>',
55634                   "</div>",
55635                   '<div class="x-grid-bottombar"></div>',
55636                  
55637                   '<div class="x-grid-resize-proxy">&#160;</div>',
55638                "</div>"
55639             );
55640             tpls.master.disableformats = true;
55641         }
55642
55643         if(!tpls.header){
55644             tpls.header = new Roo.Template(
55645                '<table border="0" cellspacing="0" cellpadding="0">',
55646                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
55647                "</table>{splits}"
55648             );
55649             tpls.header.disableformats = true;
55650         }
55651         tpls.header.compile();
55652
55653         if(!tpls.hcell){
55654             tpls.hcell = new Roo.Template(
55655                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
55656                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
55657                 "</div></td>"
55658              );
55659              tpls.hcell.disableFormats = true;
55660         }
55661         tpls.hcell.compile();
55662
55663         if(!tpls.hsplit){
55664             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
55665                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
55666             tpls.hsplit.disableFormats = true;
55667         }
55668         tpls.hsplit.compile();
55669
55670         if(!tpls.body){
55671             tpls.body = new Roo.Template(
55672                '<table border="0" cellspacing="0" cellpadding="0">',
55673                "<tbody>{rows}</tbody>",
55674                "</table>"
55675             );
55676             tpls.body.disableFormats = true;
55677         }
55678         tpls.body.compile();
55679
55680         if(!tpls.row){
55681             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
55682             tpls.row.disableFormats = true;
55683         }
55684         tpls.row.compile();
55685
55686         if(!tpls.cell){
55687             tpls.cell = new Roo.Template(
55688                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
55689                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
55690                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
55691                 "</td>"
55692             );
55693             tpls.cell.disableFormats = true;
55694         }
55695         tpls.cell.compile();
55696
55697         this.templates = tpls;
55698     },
55699
55700     // remap these for backwards compat
55701     onColWidthChange : function(){
55702         this.updateColumns.apply(this, arguments);
55703     },
55704     onHeaderChange : function(){
55705         this.updateHeaders.apply(this, arguments);
55706     }, 
55707     onHiddenChange : function(){
55708         this.handleHiddenChange.apply(this, arguments);
55709     },
55710     onColumnMove : function(){
55711         this.handleColumnMove.apply(this, arguments);
55712     },
55713     onColumnLock : function(){
55714         this.handleLockChange.apply(this, arguments);
55715     },
55716
55717     onDataChange : function(){
55718         this.refresh();
55719         this.updateHeaderSortState();
55720     },
55721
55722     onClear : function(){
55723         this.refresh();
55724     },
55725
55726     onUpdate : function(ds, record){
55727         this.refreshRow(record);
55728     },
55729
55730     refreshRow : function(record){
55731         var ds = this.ds, index;
55732         if(typeof record == 'number'){
55733             index = record;
55734             record = ds.getAt(index);
55735         }else{
55736             index = ds.indexOf(record);
55737         }
55738         this.insertRows(ds, index, index, true);
55739         this.onRemove(ds, record, index+1, true);
55740         this.syncRowHeights(index, index);
55741         this.layout();
55742         this.fireEvent("rowupdated", this, index, record);
55743     },
55744
55745     onAdd : function(ds, records, index){
55746         this.insertRows(ds, index, index + (records.length-1));
55747     },
55748
55749     onRemove : function(ds, record, index, isUpdate){
55750         if(isUpdate !== true){
55751             this.fireEvent("beforerowremoved", this, index, record);
55752         }
55753         var bt = this.getBodyTable(), lt = this.getLockedTable();
55754         if(bt.rows[index]){
55755             bt.firstChild.removeChild(bt.rows[index]);
55756         }
55757         if(lt.rows[index]){
55758             lt.firstChild.removeChild(lt.rows[index]);
55759         }
55760         if(isUpdate !== true){
55761             this.stripeRows(index);
55762             this.syncRowHeights(index, index);
55763             this.layout();
55764             this.fireEvent("rowremoved", this, index, record);
55765         }
55766     },
55767
55768     onLoad : function(){
55769         this.scrollToTop();
55770     },
55771
55772     /**
55773      * Scrolls the grid to the top
55774      */
55775     scrollToTop : function(){
55776         if(this.scroller){
55777             this.scroller.dom.scrollTop = 0;
55778             this.syncScroll();
55779         }
55780     },
55781
55782     /**
55783      * Gets a panel in the header of the grid that can be used for toolbars etc.
55784      * After modifying the contents of this panel a call to grid.autoSize() may be
55785      * required to register any changes in size.
55786      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
55787      * @return Roo.Element
55788      */
55789     getHeaderPanel : function(doShow){
55790         if(doShow){
55791             this.headerPanel.show();
55792         }
55793         return this.headerPanel;
55794     },
55795
55796     /**
55797      * Gets a panel in the footer of the grid that can be used for toolbars etc.
55798      * After modifying the contents of this panel a call to grid.autoSize() may be
55799      * required to register any changes in size.
55800      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
55801      * @return Roo.Element
55802      */
55803     getFooterPanel : function(doShow){
55804         if(doShow){
55805             this.footerPanel.show();
55806         }
55807         return this.footerPanel;
55808     },
55809
55810     initElements : function(){
55811         var E = Roo.Element;
55812         var el = this.grid.getGridEl().dom.firstChild;
55813         var cs = el.childNodes;
55814
55815         this.el = new E(el);
55816         
55817          this.focusEl = new E(el.firstChild);
55818         this.focusEl.swallowEvent("click", true);
55819         
55820         this.headerPanel = new E(cs[1]);
55821         this.headerPanel.enableDisplayMode("block");
55822
55823         this.scroller = new E(cs[2]);
55824         this.scrollSizer = new E(this.scroller.dom.firstChild);
55825
55826         this.lockedWrap = new E(cs[3]);
55827         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
55828         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
55829
55830         this.mainWrap = new E(cs[4]);
55831         this.mainHd = new E(this.mainWrap.dom.firstChild);
55832         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
55833
55834         this.footerPanel = new E(cs[5]);
55835         this.footerPanel.enableDisplayMode("block");
55836
55837         this.resizeProxy = new E(cs[6]);
55838
55839         this.headerSelector = String.format(
55840            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
55841            this.lockedHd.id, this.mainHd.id
55842         );
55843
55844         this.splitterSelector = String.format(
55845            '#{0} div.x-grid-split, #{1} div.x-grid-split',
55846            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
55847         );
55848     },
55849     idToCssName : function(s)
55850     {
55851         return s.replace(/[^a-z0-9]+/ig, '-');
55852     },
55853
55854     getHeaderCell : function(index){
55855         return Roo.DomQuery.select(this.headerSelector)[index];
55856     },
55857
55858     getHeaderCellMeasure : function(index){
55859         return this.getHeaderCell(index).firstChild;
55860     },
55861
55862     getHeaderCellText : function(index){
55863         return this.getHeaderCell(index).firstChild.firstChild;
55864     },
55865
55866     getLockedTable : function(){
55867         return this.lockedBody.dom.firstChild;
55868     },
55869
55870     getBodyTable : function(){
55871         return this.mainBody.dom.firstChild;
55872     },
55873
55874     getLockedRow : function(index){
55875         return this.getLockedTable().rows[index];
55876     },
55877
55878     getRow : function(index){
55879         return this.getBodyTable().rows[index];
55880     },
55881
55882     getRowComposite : function(index){
55883         if(!this.rowEl){
55884             this.rowEl = new Roo.CompositeElementLite();
55885         }
55886         var els = [], lrow, mrow;
55887         if(lrow = this.getLockedRow(index)){
55888             els.push(lrow);
55889         }
55890         if(mrow = this.getRow(index)){
55891             els.push(mrow);
55892         }
55893         this.rowEl.elements = els;
55894         return this.rowEl;
55895     },
55896     /**
55897      * Gets the 'td' of the cell
55898      * 
55899      * @param {Integer} rowIndex row to select
55900      * @param {Integer} colIndex column to select
55901      * 
55902      * @return {Object} 
55903      */
55904     getCell : function(rowIndex, colIndex){
55905         var locked = this.cm.getLockedCount();
55906         var source;
55907         if(colIndex < locked){
55908             source = this.lockedBody.dom.firstChild;
55909         }else{
55910             source = this.mainBody.dom.firstChild;
55911             colIndex -= locked;
55912         }
55913         return source.rows[rowIndex].childNodes[colIndex];
55914     },
55915
55916     getCellText : function(rowIndex, colIndex){
55917         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
55918     },
55919
55920     getCellBox : function(cell){
55921         var b = this.fly(cell).getBox();
55922         if(Roo.isOpera){ // opera fails to report the Y
55923             b.y = cell.offsetTop + this.mainBody.getY();
55924         }
55925         return b;
55926     },
55927
55928     getCellIndex : function(cell){
55929         var id = String(cell.className).match(this.cellRE);
55930         if(id){
55931             return parseInt(id[1], 10);
55932         }
55933         return 0;
55934     },
55935
55936     findHeaderIndex : function(n){
55937         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55938         return r ? this.getCellIndex(r) : false;
55939     },
55940
55941     findHeaderCell : function(n){
55942         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55943         return r ? r : false;
55944     },
55945
55946     findRowIndex : function(n){
55947         if(!n){
55948             return false;
55949         }
55950         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
55951         return r ? r.rowIndex : false;
55952     },
55953
55954     findCellIndex : function(node){
55955         var stop = this.el.dom;
55956         while(node && node != stop){
55957             if(this.findRE.test(node.className)){
55958                 return this.getCellIndex(node);
55959             }
55960             node = node.parentNode;
55961         }
55962         return false;
55963     },
55964
55965     getColumnId : function(index){
55966         return this.cm.getColumnId(index);
55967     },
55968
55969     getSplitters : function()
55970     {
55971         if(this.splitterSelector){
55972            return Roo.DomQuery.select(this.splitterSelector);
55973         }else{
55974             return null;
55975       }
55976     },
55977
55978     getSplitter : function(index){
55979         return this.getSplitters()[index];
55980     },
55981
55982     onRowOver : function(e, t){
55983         var row;
55984         if((row = this.findRowIndex(t)) !== false){
55985             this.getRowComposite(row).addClass("x-grid-row-over");
55986         }
55987     },
55988
55989     onRowOut : function(e, t){
55990         var row;
55991         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
55992             this.getRowComposite(row).removeClass("x-grid-row-over");
55993         }
55994     },
55995
55996     renderHeaders : function(){
55997         var cm = this.cm;
55998         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
55999         var cb = [], lb = [], sb = [], lsb = [], p = {};
56000         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56001             p.cellId = "x-grid-hd-0-" + i;
56002             p.splitId = "x-grid-csplit-0-" + i;
56003             p.id = cm.getColumnId(i);
56004             p.value = cm.getColumnHeader(i) || "";
56005             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56006             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56007             if(!cm.isLocked(i)){
56008                 cb[cb.length] = ct.apply(p);
56009                 sb[sb.length] = st.apply(p);
56010             }else{
56011                 lb[lb.length] = ct.apply(p);
56012                 lsb[lsb.length] = st.apply(p);
56013             }
56014         }
56015         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56016                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56017     },
56018
56019     updateHeaders : function(){
56020         var html = this.renderHeaders();
56021         this.lockedHd.update(html[0]);
56022         this.mainHd.update(html[1]);
56023     },
56024
56025     /**
56026      * Focuses the specified row.
56027      * @param {Number} row The row index
56028      */
56029     focusRow : function(row)
56030     {
56031         //Roo.log('GridView.focusRow');
56032         var x = this.scroller.dom.scrollLeft;
56033         this.focusCell(row, 0, false);
56034         this.scroller.dom.scrollLeft = x;
56035     },
56036
56037     /**
56038      * Focuses the specified cell.
56039      * @param {Number} row The row index
56040      * @param {Number} col The column index
56041      * @param {Boolean} hscroll false to disable horizontal scrolling
56042      */
56043     focusCell : function(row, col, hscroll)
56044     {
56045         //Roo.log('GridView.focusCell');
56046         var el = this.ensureVisible(row, col, hscroll);
56047         this.focusEl.alignTo(el, "tl-tl");
56048         if(Roo.isGecko){
56049             this.focusEl.focus();
56050         }else{
56051             this.focusEl.focus.defer(1, this.focusEl);
56052         }
56053     },
56054
56055     /**
56056      * Scrolls the specified cell into view
56057      * @param {Number} row The row index
56058      * @param {Number} col The column index
56059      * @param {Boolean} hscroll false to disable horizontal scrolling
56060      */
56061     ensureVisible : function(row, col, hscroll)
56062     {
56063         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56064         //return null; //disable for testing.
56065         if(typeof row != "number"){
56066             row = row.rowIndex;
56067         }
56068         if(row < 0 && row >= this.ds.getCount()){
56069             return  null;
56070         }
56071         col = (col !== undefined ? col : 0);
56072         var cm = this.grid.colModel;
56073         while(cm.isHidden(col)){
56074             col++;
56075         }
56076
56077         var el = this.getCell(row, col);
56078         if(!el){
56079             return null;
56080         }
56081         var c = this.scroller.dom;
56082
56083         var ctop = parseInt(el.offsetTop, 10);
56084         var cleft = parseInt(el.offsetLeft, 10);
56085         var cbot = ctop + el.offsetHeight;
56086         var cright = cleft + el.offsetWidth;
56087         
56088         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56089         var stop = parseInt(c.scrollTop, 10);
56090         var sleft = parseInt(c.scrollLeft, 10);
56091         var sbot = stop + ch;
56092         var sright = sleft + c.clientWidth;
56093         /*
56094         Roo.log('GridView.ensureVisible:' +
56095                 ' ctop:' + ctop +
56096                 ' c.clientHeight:' + c.clientHeight +
56097                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56098                 ' stop:' + stop +
56099                 ' cbot:' + cbot +
56100                 ' sbot:' + sbot +
56101                 ' ch:' + ch  
56102                 );
56103         */
56104         if(ctop < stop){
56105              c.scrollTop = ctop;
56106             //Roo.log("set scrolltop to ctop DISABLE?");
56107         }else if(cbot > sbot){
56108             //Roo.log("set scrolltop to cbot-ch");
56109             c.scrollTop = cbot-ch;
56110         }
56111         
56112         if(hscroll !== false){
56113             if(cleft < sleft){
56114                 c.scrollLeft = cleft;
56115             }else if(cright > sright){
56116                 c.scrollLeft = cright-c.clientWidth;
56117             }
56118         }
56119          
56120         return el;
56121     },
56122
56123     updateColumns : function(){
56124         this.grid.stopEditing();
56125         var cm = this.grid.colModel, colIds = this.getColumnIds();
56126         //var totalWidth = cm.getTotalWidth();
56127         var pos = 0;
56128         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56129             //if(cm.isHidden(i)) continue;
56130             var w = cm.getColumnWidth(i);
56131             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56132             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56133         }
56134         this.updateSplitters();
56135     },
56136
56137     generateRules : function(cm){
56138         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56139         Roo.util.CSS.removeStyleSheet(rulesId);
56140         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56141             var cid = cm.getColumnId(i);
56142             var align = '';
56143             if(cm.config[i].align){
56144                 align = 'text-align:'+cm.config[i].align+';';
56145             }
56146             var hidden = '';
56147             if(cm.isHidden(i)){
56148                 hidden = 'display:none;';
56149             }
56150             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56151             ruleBuf.push(
56152                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56153                     this.hdSelector, cid, " {\n", align, width, "}\n",
56154                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56155                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56156         }
56157         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56158     },
56159
56160     updateSplitters : function(){
56161         var cm = this.cm, s = this.getSplitters();
56162         if(s){ // splitters not created yet
56163             var pos = 0, locked = true;
56164             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56165                 if(cm.isHidden(i)) {
56166                     continue;
56167                 }
56168                 var w = cm.getColumnWidth(i); // make sure it's a number
56169                 if(!cm.isLocked(i) && locked){
56170                     pos = 0;
56171                     locked = false;
56172                 }
56173                 pos += w;
56174                 s[i].style.left = (pos-this.splitOffset) + "px";
56175             }
56176         }
56177     },
56178
56179     handleHiddenChange : function(colModel, colIndex, hidden){
56180         if(hidden){
56181             this.hideColumn(colIndex);
56182         }else{
56183             this.unhideColumn(colIndex);
56184         }
56185     },
56186
56187     hideColumn : function(colIndex){
56188         var cid = this.getColumnId(colIndex);
56189         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56190         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56191         if(Roo.isSafari){
56192             this.updateHeaders();
56193         }
56194         this.updateSplitters();
56195         this.layout();
56196     },
56197
56198     unhideColumn : function(colIndex){
56199         var cid = this.getColumnId(colIndex);
56200         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56201         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56202
56203         if(Roo.isSafari){
56204             this.updateHeaders();
56205         }
56206         this.updateSplitters();
56207         this.layout();
56208     },
56209
56210     insertRows : function(dm, firstRow, lastRow, isUpdate){
56211         if(firstRow == 0 && lastRow == dm.getCount()-1){
56212             this.refresh();
56213         }else{
56214             if(!isUpdate){
56215                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56216             }
56217             var s = this.getScrollState();
56218             var markup = this.renderRows(firstRow, lastRow);
56219             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56220             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56221             this.restoreScroll(s);
56222             if(!isUpdate){
56223                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56224                 this.syncRowHeights(firstRow, lastRow);
56225                 this.stripeRows(firstRow);
56226                 this.layout();
56227             }
56228         }
56229     },
56230
56231     bufferRows : function(markup, target, index){
56232         var before = null, trows = target.rows, tbody = target.tBodies[0];
56233         if(index < trows.length){
56234             before = trows[index];
56235         }
56236         var b = document.createElement("div");
56237         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56238         var rows = b.firstChild.rows;
56239         for(var i = 0, len = rows.length; i < len; i++){
56240             if(before){
56241                 tbody.insertBefore(rows[0], before);
56242             }else{
56243                 tbody.appendChild(rows[0]);
56244             }
56245         }
56246         b.innerHTML = "";
56247         b = null;
56248     },
56249
56250     deleteRows : function(dm, firstRow, lastRow){
56251         if(dm.getRowCount()<1){
56252             this.fireEvent("beforerefresh", this);
56253             this.mainBody.update("");
56254             this.lockedBody.update("");
56255             this.fireEvent("refresh", this);
56256         }else{
56257             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56258             var bt = this.getBodyTable();
56259             var tbody = bt.firstChild;
56260             var rows = bt.rows;
56261             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56262                 tbody.removeChild(rows[firstRow]);
56263             }
56264             this.stripeRows(firstRow);
56265             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56266         }
56267     },
56268
56269     updateRows : function(dataSource, firstRow, lastRow){
56270         var s = this.getScrollState();
56271         this.refresh();
56272         this.restoreScroll(s);
56273     },
56274
56275     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56276         if(!noRefresh){
56277            this.refresh();
56278         }
56279         this.updateHeaderSortState();
56280     },
56281
56282     getScrollState : function(){
56283         
56284         var sb = this.scroller.dom;
56285         return {left: sb.scrollLeft, top: sb.scrollTop};
56286     },
56287
56288     stripeRows : function(startRow){
56289         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56290             return;
56291         }
56292         startRow = startRow || 0;
56293         var rows = this.getBodyTable().rows;
56294         var lrows = this.getLockedTable().rows;
56295         var cls = ' x-grid-row-alt ';
56296         for(var i = startRow, len = rows.length; i < len; i++){
56297             var row = rows[i], lrow = lrows[i];
56298             var isAlt = ((i+1) % 2 == 0);
56299             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56300             if(isAlt == hasAlt){
56301                 continue;
56302             }
56303             if(isAlt){
56304                 row.className += " x-grid-row-alt";
56305             }else{
56306                 row.className = row.className.replace("x-grid-row-alt", "");
56307             }
56308             if(lrow){
56309                 lrow.className = row.className;
56310             }
56311         }
56312     },
56313
56314     restoreScroll : function(state){
56315         //Roo.log('GridView.restoreScroll');
56316         var sb = this.scroller.dom;
56317         sb.scrollLeft = state.left;
56318         sb.scrollTop = state.top;
56319         this.syncScroll();
56320     },
56321
56322     syncScroll : function(){
56323         //Roo.log('GridView.syncScroll');
56324         var sb = this.scroller.dom;
56325         var sh = this.mainHd.dom;
56326         var bs = this.mainBody.dom;
56327         var lv = this.lockedBody.dom;
56328         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56329         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56330     },
56331
56332     handleScroll : function(e){
56333         this.syncScroll();
56334         var sb = this.scroller.dom;
56335         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56336         e.stopEvent();
56337     },
56338
56339     handleWheel : function(e){
56340         var d = e.getWheelDelta();
56341         this.scroller.dom.scrollTop -= d*22;
56342         // set this here to prevent jumpy scrolling on large tables
56343         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56344         e.stopEvent();
56345     },
56346
56347     renderRows : function(startRow, endRow){
56348         // pull in all the crap needed to render rows
56349         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56350         var colCount = cm.getColumnCount();
56351
56352         if(ds.getCount() < 1){
56353             return ["", ""];
56354         }
56355
56356         // build a map for all the columns
56357         var cs = [];
56358         for(var i = 0; i < colCount; i++){
56359             var name = cm.getDataIndex(i);
56360             cs[i] = {
56361                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56362                 renderer : cm.getRenderer(i),
56363                 id : cm.getColumnId(i),
56364                 locked : cm.isLocked(i),
56365                 has_editor : cm.isCellEditable(i)
56366             };
56367         }
56368
56369         startRow = startRow || 0;
56370         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56371
56372         // records to render
56373         var rs = ds.getRange(startRow, endRow);
56374
56375         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56376     },
56377
56378     // As much as I hate to duplicate code, this was branched because FireFox really hates
56379     // [].join("") on strings. The performance difference was substantial enough to
56380     // branch this function
56381     doRender : Roo.isGecko ?
56382             function(cs, rs, ds, startRow, colCount, stripe){
56383                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56384                 // buffers
56385                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56386                 
56387                 var hasListener = this.grid.hasListener('rowclass');
56388                 var rowcfg = {};
56389                 for(var j = 0, len = rs.length; j < len; j++){
56390                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56391                     for(var i = 0; i < colCount; i++){
56392                         c = cs[i];
56393                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56394                         p.id = c.id;
56395                         p.css = p.attr = "";
56396                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56397                         if(p.value == undefined || p.value === "") {
56398                             p.value = "&#160;";
56399                         }
56400                         if(c.has_editor){
56401                             p.css += ' x-grid-editable-cell';
56402                         }
56403                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56404                             p.css +=  ' x-grid-dirty-cell';
56405                         }
56406                         var markup = ct.apply(p);
56407                         if(!c.locked){
56408                             cb+= markup;
56409                         }else{
56410                             lcb+= markup;
56411                         }
56412                     }
56413                     var alt = [];
56414                     if(stripe && ((rowIndex+1) % 2 == 0)){
56415                         alt.push("x-grid-row-alt")
56416                     }
56417                     if(r.dirty){
56418                         alt.push(  " x-grid-dirty-row");
56419                     }
56420                     rp.cells = lcb;
56421                     if(this.getRowClass){
56422                         alt.push(this.getRowClass(r, rowIndex));
56423                     }
56424                     if (hasListener) {
56425                         rowcfg = {
56426                              
56427                             record: r,
56428                             rowIndex : rowIndex,
56429                             rowClass : ''
56430                         };
56431                         this.grid.fireEvent('rowclass', this, rowcfg);
56432                         alt.push(rowcfg.rowClass);
56433                     }
56434                     rp.alt = alt.join(" ");
56435                     lbuf+= rt.apply(rp);
56436                     rp.cells = cb;
56437                     buf+=  rt.apply(rp);
56438                 }
56439                 return [lbuf, buf];
56440             } :
56441             function(cs, rs, ds, startRow, colCount, stripe){
56442                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56443                 // buffers
56444                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56445                 var hasListener = this.grid.hasListener('rowclass');
56446  
56447                 var rowcfg = {};
56448                 for(var j = 0, len = rs.length; j < len; j++){
56449                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56450                     for(var i = 0; i < colCount; i++){
56451                         c = cs[i];
56452                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56453                         p.id = c.id;
56454                         p.css = p.attr = "";
56455                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56456                         if(p.value == undefined || p.value === "") {
56457                             p.value = "&#160;";
56458                         }
56459                         //Roo.log(c);
56460                          if(c.has_editor){
56461                             p.css += ' x-grid-editable-cell';
56462                         }
56463                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56464                             p.css += ' x-grid-dirty-cell' 
56465                         }
56466                         
56467                         var markup = ct.apply(p);
56468                         if(!c.locked){
56469                             cb[cb.length] = markup;
56470                         }else{
56471                             lcb[lcb.length] = markup;
56472                         }
56473                     }
56474                     var alt = [];
56475                     if(stripe && ((rowIndex+1) % 2 == 0)){
56476                         alt.push( "x-grid-row-alt");
56477                     }
56478                     if(r.dirty){
56479                         alt.push(" x-grid-dirty-row");
56480                     }
56481                     rp.cells = lcb;
56482                     if(this.getRowClass){
56483                         alt.push( this.getRowClass(r, rowIndex));
56484                     }
56485                     if (hasListener) {
56486                         rowcfg = {
56487                              
56488                             record: r,
56489                             rowIndex : rowIndex,
56490                             rowClass : ''
56491                         };
56492                         this.grid.fireEvent('rowclass', this, rowcfg);
56493                         alt.push(rowcfg.rowClass);
56494                     }
56495                     
56496                     rp.alt = alt.join(" ");
56497                     rp.cells = lcb.join("");
56498                     lbuf[lbuf.length] = rt.apply(rp);
56499                     rp.cells = cb.join("");
56500                     buf[buf.length] =  rt.apply(rp);
56501                 }
56502                 return [lbuf.join(""), buf.join("")];
56503             },
56504
56505     renderBody : function(){
56506         var markup = this.renderRows();
56507         var bt = this.templates.body;
56508         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56509     },
56510
56511     /**
56512      * Refreshes the grid
56513      * @param {Boolean} headersToo
56514      */
56515     refresh : function(headersToo){
56516         this.fireEvent("beforerefresh", this);
56517         this.grid.stopEditing();
56518         var result = this.renderBody();
56519         this.lockedBody.update(result[0]);
56520         this.mainBody.update(result[1]);
56521         if(headersToo === true){
56522             this.updateHeaders();
56523             this.updateColumns();
56524             this.updateSplitters();
56525             this.updateHeaderSortState();
56526         }
56527         this.syncRowHeights();
56528         this.layout();
56529         this.fireEvent("refresh", this);
56530     },
56531
56532     handleColumnMove : function(cm, oldIndex, newIndex){
56533         this.indexMap = null;
56534         var s = this.getScrollState();
56535         this.refresh(true);
56536         this.restoreScroll(s);
56537         this.afterMove(newIndex);
56538     },
56539
56540     afterMove : function(colIndex){
56541         if(this.enableMoveAnim && Roo.enableFx){
56542             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56543         }
56544         // if multisort - fix sortOrder, and reload..
56545         if (this.grid.dataSource.multiSort) {
56546             // the we can call sort again..
56547             var dm = this.grid.dataSource;
56548             var cm = this.grid.colModel;
56549             var so = [];
56550             for(var i = 0; i < cm.config.length; i++ ) {
56551                 
56552                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56553                     continue; // dont' bother, it's not in sort list or being set.
56554                 }
56555                 
56556                 so.push(cm.config[i].dataIndex);
56557             };
56558             dm.sortOrder = so;
56559             dm.load(dm.lastOptions);
56560             
56561             
56562         }
56563         
56564     },
56565
56566     updateCell : function(dm, rowIndex, dataIndex){
56567         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56568         if(typeof colIndex == "undefined"){ // not present in grid
56569             return;
56570         }
56571         var cm = this.grid.colModel;
56572         var cell = this.getCell(rowIndex, colIndex);
56573         var cellText = this.getCellText(rowIndex, colIndex);
56574
56575         var p = {
56576             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56577             id : cm.getColumnId(colIndex),
56578             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56579         };
56580         var renderer = cm.getRenderer(colIndex);
56581         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56582         if(typeof val == "undefined" || val === "") {
56583             val = "&#160;";
56584         }
56585         cellText.innerHTML = val;
56586         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56587         this.syncRowHeights(rowIndex, rowIndex);
56588     },
56589
56590     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56591         var maxWidth = 0;
56592         if(this.grid.autoSizeHeaders){
56593             var h = this.getHeaderCellMeasure(colIndex);
56594             maxWidth = Math.max(maxWidth, h.scrollWidth);
56595         }
56596         var tb, index;
56597         if(this.cm.isLocked(colIndex)){
56598             tb = this.getLockedTable();
56599             index = colIndex;
56600         }else{
56601             tb = this.getBodyTable();
56602             index = colIndex - this.cm.getLockedCount();
56603         }
56604         if(tb && tb.rows){
56605             var rows = tb.rows;
56606             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
56607             for(var i = 0; i < stopIndex; i++){
56608                 var cell = rows[i].childNodes[index].firstChild;
56609                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
56610             }
56611         }
56612         return maxWidth + /*margin for error in IE*/ 5;
56613     },
56614     /**
56615      * Autofit a column to its content.
56616      * @param {Number} colIndex
56617      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
56618      */
56619      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
56620          if(this.cm.isHidden(colIndex)){
56621              return; // can't calc a hidden column
56622          }
56623         if(forceMinSize){
56624             var cid = this.cm.getColumnId(colIndex);
56625             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
56626            if(this.grid.autoSizeHeaders){
56627                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
56628            }
56629         }
56630         var newWidth = this.calcColumnWidth(colIndex);
56631         this.cm.setColumnWidth(colIndex,
56632             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
56633         if(!suppressEvent){
56634             this.grid.fireEvent("columnresize", colIndex, newWidth);
56635         }
56636     },
56637
56638     /**
56639      * Autofits all columns to their content and then expands to fit any extra space in the grid
56640      */
56641      autoSizeColumns : function(){
56642         var cm = this.grid.colModel;
56643         var colCount = cm.getColumnCount();
56644         for(var i = 0; i < colCount; i++){
56645             this.autoSizeColumn(i, true, true);
56646         }
56647         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
56648             this.fitColumns();
56649         }else{
56650             this.updateColumns();
56651             this.layout();
56652         }
56653     },
56654
56655     /**
56656      * Autofits all columns to the grid's width proportionate with their current size
56657      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
56658      */
56659     fitColumns : function(reserveScrollSpace){
56660         var cm = this.grid.colModel;
56661         var colCount = cm.getColumnCount();
56662         var cols = [];
56663         var width = 0;
56664         var i, w;
56665         for (i = 0; i < colCount; i++){
56666             if(!cm.isHidden(i) && !cm.isFixed(i)){
56667                 w = cm.getColumnWidth(i);
56668                 cols.push(i);
56669                 cols.push(w);
56670                 width += w;
56671             }
56672         }
56673         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
56674         if(reserveScrollSpace){
56675             avail -= 17;
56676         }
56677         var frac = (avail - cm.getTotalWidth())/width;
56678         while (cols.length){
56679             w = cols.pop();
56680             i = cols.pop();
56681             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
56682         }
56683         this.updateColumns();
56684         this.layout();
56685     },
56686
56687     onRowSelect : function(rowIndex){
56688         var row = this.getRowComposite(rowIndex);
56689         row.addClass("x-grid-row-selected");
56690     },
56691
56692     onRowDeselect : function(rowIndex){
56693         var row = this.getRowComposite(rowIndex);
56694         row.removeClass("x-grid-row-selected");
56695     },
56696
56697     onCellSelect : function(row, col){
56698         var cell = this.getCell(row, col);
56699         if(cell){
56700             Roo.fly(cell).addClass("x-grid-cell-selected");
56701         }
56702     },
56703
56704     onCellDeselect : function(row, col){
56705         var cell = this.getCell(row, col);
56706         if(cell){
56707             Roo.fly(cell).removeClass("x-grid-cell-selected");
56708         }
56709     },
56710
56711     updateHeaderSortState : function(){
56712         
56713         // sort state can be single { field: xxx, direction : yyy}
56714         // or   { xxx=>ASC , yyy : DESC ..... }
56715         
56716         var mstate = {};
56717         if (!this.ds.multiSort) { 
56718             var state = this.ds.getSortState();
56719             if(!state){
56720                 return;
56721             }
56722             mstate[state.field] = state.direction;
56723             // FIXME... - this is not used here.. but might be elsewhere..
56724             this.sortState = state;
56725             
56726         } else {
56727             mstate = this.ds.sortToggle;
56728         }
56729         //remove existing sort classes..
56730         
56731         var sc = this.sortClasses;
56732         var hds = this.el.select(this.headerSelector).removeClass(sc);
56733         
56734         for(var f in mstate) {
56735         
56736             var sortColumn = this.cm.findColumnIndex(f);
56737             
56738             if(sortColumn != -1){
56739                 var sortDir = mstate[f];        
56740                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
56741             }
56742         }
56743         
56744          
56745         
56746     },
56747
56748
56749     handleHeaderClick : function(g, index,e){
56750         
56751         Roo.log("header click");
56752         
56753         if (Roo.isTouch) {
56754             // touch events on header are handled by context
56755             this.handleHdCtx(g,index,e);
56756             return;
56757         }
56758         
56759         
56760         if(this.headersDisabled){
56761             return;
56762         }
56763         var dm = g.dataSource, cm = g.colModel;
56764         if(!cm.isSortable(index)){
56765             return;
56766         }
56767         g.stopEditing();
56768         
56769         if (dm.multiSort) {
56770             // update the sortOrder
56771             var so = [];
56772             for(var i = 0; i < cm.config.length; i++ ) {
56773                 
56774                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
56775                     continue; // dont' bother, it's not in sort list or being set.
56776                 }
56777                 
56778                 so.push(cm.config[i].dataIndex);
56779             };
56780             dm.sortOrder = so;
56781         }
56782         
56783         
56784         dm.sort(cm.getDataIndex(index));
56785     },
56786
56787
56788     destroy : function(){
56789         if(this.colMenu){
56790             this.colMenu.removeAll();
56791             Roo.menu.MenuMgr.unregister(this.colMenu);
56792             this.colMenu.getEl().remove();
56793             delete this.colMenu;
56794         }
56795         if(this.hmenu){
56796             this.hmenu.removeAll();
56797             Roo.menu.MenuMgr.unregister(this.hmenu);
56798             this.hmenu.getEl().remove();
56799             delete this.hmenu;
56800         }
56801         if(this.grid.enableColumnMove){
56802             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56803             if(dds){
56804                 for(var dd in dds){
56805                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
56806                         var elid = dds[dd].dragElId;
56807                         dds[dd].unreg();
56808                         Roo.get(elid).remove();
56809                     } else if(dds[dd].config.isTarget){
56810                         dds[dd].proxyTop.remove();
56811                         dds[dd].proxyBottom.remove();
56812                         dds[dd].unreg();
56813                     }
56814                     if(Roo.dd.DDM.locationCache[dd]){
56815                         delete Roo.dd.DDM.locationCache[dd];
56816                     }
56817                 }
56818                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56819             }
56820         }
56821         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
56822         this.bind(null, null);
56823         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
56824     },
56825
56826     handleLockChange : function(){
56827         this.refresh(true);
56828     },
56829
56830     onDenyColumnLock : function(){
56831
56832     },
56833
56834     onDenyColumnHide : function(){
56835
56836     },
56837
56838     handleHdMenuClick : function(item){
56839         var index = this.hdCtxIndex;
56840         var cm = this.cm, ds = this.ds;
56841         switch(item.id){
56842             case "asc":
56843                 ds.sort(cm.getDataIndex(index), "ASC");
56844                 break;
56845             case "desc":
56846                 ds.sort(cm.getDataIndex(index), "DESC");
56847                 break;
56848             case "lock":
56849                 var lc = cm.getLockedCount();
56850                 if(cm.getColumnCount(true) <= lc+1){
56851                     this.onDenyColumnLock();
56852                     return;
56853                 }
56854                 if(lc != index){
56855                     cm.setLocked(index, true, true);
56856                     cm.moveColumn(index, lc);
56857                     this.grid.fireEvent("columnmove", index, lc);
56858                 }else{
56859                     cm.setLocked(index, true);
56860                 }
56861             break;
56862             case "unlock":
56863                 var lc = cm.getLockedCount();
56864                 if((lc-1) != index){
56865                     cm.setLocked(index, false, true);
56866                     cm.moveColumn(index, lc-1);
56867                     this.grid.fireEvent("columnmove", index, lc-1);
56868                 }else{
56869                     cm.setLocked(index, false);
56870                 }
56871             break;
56872             case 'wider': // used to expand cols on touch..
56873             case 'narrow':
56874                 var cw = cm.getColumnWidth(index);
56875                 cw += (item.id == 'wider' ? 1 : -1) * 50;
56876                 cw = Math.max(0, cw);
56877                 cw = Math.min(cw,4000);
56878                 cm.setColumnWidth(index, cw);
56879                 break;
56880                 
56881             default:
56882                 index = cm.getIndexById(item.id.substr(4));
56883                 if(index != -1){
56884                     if(item.checked && cm.getColumnCount(true) <= 1){
56885                         this.onDenyColumnHide();
56886                         return false;
56887                     }
56888                     cm.setHidden(index, item.checked);
56889                 }
56890         }
56891         return true;
56892     },
56893
56894     beforeColMenuShow : function(){
56895         var cm = this.cm,  colCount = cm.getColumnCount();
56896         this.colMenu.removeAll();
56897         for(var i = 0; i < colCount; i++){
56898             this.colMenu.add(new Roo.menu.CheckItem({
56899                 id: "col-"+cm.getColumnId(i),
56900                 text: cm.getColumnHeader(i),
56901                 checked: !cm.isHidden(i),
56902                 hideOnClick:false
56903             }));
56904         }
56905     },
56906
56907     handleHdCtx : function(g, index, e){
56908         e.stopEvent();
56909         var hd = this.getHeaderCell(index);
56910         this.hdCtxIndex = index;
56911         var ms = this.hmenu.items, cm = this.cm;
56912         ms.get("asc").setDisabled(!cm.isSortable(index));
56913         ms.get("desc").setDisabled(!cm.isSortable(index));
56914         if(this.grid.enableColLock !== false){
56915             ms.get("lock").setDisabled(cm.isLocked(index));
56916             ms.get("unlock").setDisabled(!cm.isLocked(index));
56917         }
56918         this.hmenu.show(hd, "tl-bl");
56919     },
56920
56921     handleHdOver : function(e){
56922         var hd = this.findHeaderCell(e.getTarget());
56923         if(hd && !this.headersDisabled){
56924             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
56925                this.fly(hd).addClass("x-grid-hd-over");
56926             }
56927         }
56928     },
56929
56930     handleHdOut : function(e){
56931         var hd = this.findHeaderCell(e.getTarget());
56932         if(hd){
56933             this.fly(hd).removeClass("x-grid-hd-over");
56934         }
56935     },
56936
56937     handleSplitDblClick : function(e, t){
56938         var i = this.getCellIndex(t);
56939         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
56940             this.autoSizeColumn(i, true);
56941             this.layout();
56942         }
56943     },
56944
56945     render : function(){
56946
56947         var cm = this.cm;
56948         var colCount = cm.getColumnCount();
56949
56950         if(this.grid.monitorWindowResize === true){
56951             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
56952         }
56953         var header = this.renderHeaders();
56954         var body = this.templates.body.apply({rows:""});
56955         var html = this.templates.master.apply({
56956             lockedBody: body,
56957             body: body,
56958             lockedHeader: header[0],
56959             header: header[1]
56960         });
56961
56962         //this.updateColumns();
56963
56964         this.grid.getGridEl().dom.innerHTML = html;
56965
56966         this.initElements();
56967         
56968         // a kludge to fix the random scolling effect in webkit
56969         this.el.on("scroll", function() {
56970             this.el.dom.scrollTop=0; // hopefully not recursive..
56971         },this);
56972
56973         this.scroller.on("scroll", this.handleScroll, this);
56974         this.lockedBody.on("mousewheel", this.handleWheel, this);
56975         this.mainBody.on("mousewheel", this.handleWheel, this);
56976
56977         this.mainHd.on("mouseover", this.handleHdOver, this);
56978         this.mainHd.on("mouseout", this.handleHdOut, this);
56979         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
56980                 {delegate: "."+this.splitClass});
56981
56982         this.lockedHd.on("mouseover", this.handleHdOver, this);
56983         this.lockedHd.on("mouseout", this.handleHdOut, this);
56984         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
56985                 {delegate: "."+this.splitClass});
56986
56987         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
56988             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56989         }
56990
56991         this.updateSplitters();
56992
56993         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
56994             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56995             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56996         }
56997
56998         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
56999             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57000             this.hmenu.add(
57001                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57002                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57003             );
57004             if(this.grid.enableColLock !== false){
57005                 this.hmenu.add('-',
57006                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57007                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57008                 );
57009             }
57010             if (Roo.isTouch) {
57011                  this.hmenu.add('-',
57012                     {id:"wider", text: this.columnsWiderText},
57013                     {id:"narrow", text: this.columnsNarrowText }
57014                 );
57015                 
57016                  
57017             }
57018             
57019             if(this.grid.enableColumnHide !== false){
57020
57021                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57022                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57023                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57024
57025                 this.hmenu.add('-',
57026                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57027                 );
57028             }
57029             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57030
57031             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57032         }
57033
57034         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57035             this.dd = new Roo.grid.GridDragZone(this.grid, {
57036                 ddGroup : this.grid.ddGroup || 'GridDD'
57037             });
57038             
57039         }
57040
57041         /*
57042         for(var i = 0; i < colCount; i++){
57043             if(cm.isHidden(i)){
57044                 this.hideColumn(i);
57045             }
57046             if(cm.config[i].align){
57047                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57048                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57049             }
57050         }*/
57051         
57052         this.updateHeaderSortState();
57053
57054         this.beforeInitialResize();
57055         this.layout(true);
57056
57057         // two part rendering gives faster view to the user
57058         this.renderPhase2.defer(1, this);
57059     },
57060
57061     renderPhase2 : function(){
57062         // render the rows now
57063         this.refresh();
57064         if(this.grid.autoSizeColumns){
57065             this.autoSizeColumns();
57066         }
57067     },
57068
57069     beforeInitialResize : function(){
57070
57071     },
57072
57073     onColumnSplitterMoved : function(i, w){
57074         this.userResized = true;
57075         var cm = this.grid.colModel;
57076         cm.setColumnWidth(i, w, true);
57077         var cid = cm.getColumnId(i);
57078         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57079         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57080         this.updateSplitters();
57081         this.layout();
57082         this.grid.fireEvent("columnresize", i, w);
57083     },
57084
57085     syncRowHeights : function(startIndex, endIndex){
57086         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57087             startIndex = startIndex || 0;
57088             var mrows = this.getBodyTable().rows;
57089             var lrows = this.getLockedTable().rows;
57090             var len = mrows.length-1;
57091             endIndex = Math.min(endIndex || len, len);
57092             for(var i = startIndex; i <= endIndex; i++){
57093                 var m = mrows[i], l = lrows[i];
57094                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57095                 m.style.height = l.style.height = h + "px";
57096             }
57097         }
57098     },
57099
57100     layout : function(initialRender, is2ndPass){
57101         var g = this.grid;
57102         var auto = g.autoHeight;
57103         var scrollOffset = 16;
57104         var c = g.getGridEl(), cm = this.cm,
57105                 expandCol = g.autoExpandColumn,
57106                 gv = this;
57107         //c.beginMeasure();
57108
57109         if(!c.dom.offsetWidth){ // display:none?
57110             if(initialRender){
57111                 this.lockedWrap.show();
57112                 this.mainWrap.show();
57113             }
57114             return;
57115         }
57116
57117         var hasLock = this.cm.isLocked(0);
57118
57119         var tbh = this.headerPanel.getHeight();
57120         var bbh = this.footerPanel.getHeight();
57121
57122         if(auto){
57123             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57124             var newHeight = ch + c.getBorderWidth("tb");
57125             if(g.maxHeight){
57126                 newHeight = Math.min(g.maxHeight, newHeight);
57127             }
57128             c.setHeight(newHeight);
57129         }
57130
57131         if(g.autoWidth){
57132             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57133         }
57134
57135         var s = this.scroller;
57136
57137         var csize = c.getSize(true);
57138
57139         this.el.setSize(csize.width, csize.height);
57140
57141         this.headerPanel.setWidth(csize.width);
57142         this.footerPanel.setWidth(csize.width);
57143
57144         var hdHeight = this.mainHd.getHeight();
57145         var vw = csize.width;
57146         var vh = csize.height - (tbh + bbh);
57147
57148         s.setSize(vw, vh);
57149
57150         var bt = this.getBodyTable();
57151         
57152         if(cm.getLockedCount() == cm.config.length){
57153             bt = this.getLockedTable();
57154         }
57155         
57156         var ltWidth = hasLock ?
57157                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57158
57159         var scrollHeight = bt.offsetHeight;
57160         var scrollWidth = ltWidth + bt.offsetWidth;
57161         var vscroll = false, hscroll = false;
57162
57163         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57164
57165         var lw = this.lockedWrap, mw = this.mainWrap;
57166         var lb = this.lockedBody, mb = this.mainBody;
57167
57168         setTimeout(function(){
57169             var t = s.dom.offsetTop;
57170             var w = s.dom.clientWidth,
57171                 h = s.dom.clientHeight;
57172
57173             lw.setTop(t);
57174             lw.setSize(ltWidth, h);
57175
57176             mw.setLeftTop(ltWidth, t);
57177             mw.setSize(w-ltWidth, h);
57178
57179             lb.setHeight(h-hdHeight);
57180             mb.setHeight(h-hdHeight);
57181
57182             if(is2ndPass !== true && !gv.userResized && expandCol){
57183                 // high speed resize without full column calculation
57184                 
57185                 var ci = cm.getIndexById(expandCol);
57186                 if (ci < 0) {
57187                     ci = cm.findColumnIndex(expandCol);
57188                 }
57189                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57190                 var expandId = cm.getColumnId(ci);
57191                 var  tw = cm.getTotalWidth(false);
57192                 var currentWidth = cm.getColumnWidth(ci);
57193                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57194                 if(currentWidth != cw){
57195                     cm.setColumnWidth(ci, cw, true);
57196                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57197                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57198                     gv.updateSplitters();
57199                     gv.layout(false, true);
57200                 }
57201             }
57202
57203             if(initialRender){
57204                 lw.show();
57205                 mw.show();
57206             }
57207             //c.endMeasure();
57208         }, 10);
57209     },
57210
57211     onWindowResize : function(){
57212         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57213             return;
57214         }
57215         this.layout();
57216     },
57217
57218     appendFooter : function(parentEl){
57219         return null;
57220     },
57221
57222     sortAscText : "Sort Ascending",
57223     sortDescText : "Sort Descending",
57224     lockText : "Lock Column",
57225     unlockText : "Unlock Column",
57226     columnsText : "Columns",
57227  
57228     columnsWiderText : "Wider",
57229     columnsNarrowText : "Thinner"
57230 });
57231
57232
57233 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57234     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57235     this.proxy.el.addClass('x-grid3-col-dd');
57236 };
57237
57238 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57239     handleMouseDown : function(e){
57240
57241     },
57242
57243     callHandleMouseDown : function(e){
57244         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57245     }
57246 });
57247 /*
57248  * Based on:
57249  * Ext JS Library 1.1.1
57250  * Copyright(c) 2006-2007, Ext JS, LLC.
57251  *
57252  * Originally Released Under LGPL - original licence link has changed is not relivant.
57253  *
57254  * Fork - LGPL
57255  * <script type="text/javascript">
57256  */
57257  
57258 // private
57259 // This is a support class used internally by the Grid components
57260 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57261     this.grid = grid;
57262     this.view = grid.getView();
57263     this.proxy = this.view.resizeProxy;
57264     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57265         "gridSplitters" + this.grid.getGridEl().id, {
57266         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57267     });
57268     this.setHandleElId(Roo.id(hd));
57269     this.setOuterHandleElId(Roo.id(hd2));
57270     this.scroll = false;
57271 };
57272 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57273     fly: Roo.Element.fly,
57274
57275     b4StartDrag : function(x, y){
57276         this.view.headersDisabled = true;
57277         this.proxy.setHeight(this.view.mainWrap.getHeight());
57278         var w = this.cm.getColumnWidth(this.cellIndex);
57279         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57280         this.resetConstraints();
57281         this.setXConstraint(minw, 1000);
57282         this.setYConstraint(0, 0);
57283         this.minX = x - minw;
57284         this.maxX = x + 1000;
57285         this.startPos = x;
57286         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57287     },
57288
57289
57290     handleMouseDown : function(e){
57291         ev = Roo.EventObject.setEvent(e);
57292         var t = this.fly(ev.getTarget());
57293         if(t.hasClass("x-grid-split")){
57294             this.cellIndex = this.view.getCellIndex(t.dom);
57295             this.split = t.dom;
57296             this.cm = this.grid.colModel;
57297             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57298                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57299             }
57300         }
57301     },
57302
57303     endDrag : function(e){
57304         this.view.headersDisabled = false;
57305         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57306         var diff = endX - this.startPos;
57307         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57308     },
57309
57310     autoOffset : function(){
57311         this.setDelta(0,0);
57312     }
57313 });/*
57314  * Based on:
57315  * Ext JS Library 1.1.1
57316  * Copyright(c) 2006-2007, Ext JS, LLC.
57317  *
57318  * Originally Released Under LGPL - original licence link has changed is not relivant.
57319  *
57320  * Fork - LGPL
57321  * <script type="text/javascript">
57322  */
57323  
57324 // private
57325 // This is a support class used internally by the Grid components
57326 Roo.grid.GridDragZone = function(grid, config){
57327     this.view = grid.getView();
57328     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57329     if(this.view.lockedBody){
57330         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57331         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57332     }
57333     this.scroll = false;
57334     this.grid = grid;
57335     this.ddel = document.createElement('div');
57336     this.ddel.className = 'x-grid-dd-wrap';
57337 };
57338
57339 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57340     ddGroup : "GridDD",
57341
57342     getDragData : function(e){
57343         var t = Roo.lib.Event.getTarget(e);
57344         var rowIndex = this.view.findRowIndex(t);
57345         var sm = this.grid.selModel;
57346             
57347         //Roo.log(rowIndex);
57348         
57349         if (sm.getSelectedCell) {
57350             // cell selection..
57351             if (!sm.getSelectedCell()) {
57352                 return false;
57353             }
57354             if (rowIndex != sm.getSelectedCell()[0]) {
57355                 return false;
57356             }
57357         
57358         }
57359         
57360         if(rowIndex !== false){
57361             
57362             // if editorgrid.. 
57363             
57364             
57365             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57366                
57367             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57368               //  
57369             //}
57370             if (e.hasModifier()){
57371                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57372             }
57373             
57374             Roo.log("getDragData");
57375             
57376             return {
57377                 grid: this.grid,
57378                 ddel: this.ddel,
57379                 rowIndex: rowIndex,
57380                 selections:sm.getSelections ? sm.getSelections() : (
57381                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57382                 )
57383             };
57384         }
57385         return false;
57386     },
57387
57388     onInitDrag : function(e){
57389         var data = this.dragData;
57390         this.ddel.innerHTML = this.grid.getDragDropText();
57391         this.proxy.update(this.ddel);
57392         // fire start drag?
57393     },
57394
57395     afterRepair : function(){
57396         this.dragging = false;
57397     },
57398
57399     getRepairXY : function(e, data){
57400         return false;
57401     },
57402
57403     onEndDrag : function(data, e){
57404         // fire end drag?
57405     },
57406
57407     onValidDrop : function(dd, e, id){
57408         // fire drag drop?
57409         this.hideProxy();
57410     },
57411
57412     beforeInvalidDrop : function(e, id){
57413
57414     }
57415 });/*
57416  * Based on:
57417  * Ext JS Library 1.1.1
57418  * Copyright(c) 2006-2007, Ext JS, LLC.
57419  *
57420  * Originally Released Under LGPL - original licence link has changed is not relivant.
57421  *
57422  * Fork - LGPL
57423  * <script type="text/javascript">
57424  */
57425  
57426
57427 /**
57428  * @class Roo.grid.ColumnModel
57429  * @extends Roo.util.Observable
57430  * This is the default implementation of a ColumnModel used by the Grid. It defines
57431  * the columns in the grid.
57432  * <br>Usage:<br>
57433  <pre><code>
57434  var colModel = new Roo.grid.ColumnModel([
57435         {header: "Ticker", width: 60, sortable: true, locked: true},
57436         {header: "Company Name", width: 150, sortable: true},
57437         {header: "Market Cap.", width: 100, sortable: true},
57438         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57439         {header: "Employees", width: 100, sortable: true, resizable: false}
57440  ]);
57441  </code></pre>
57442  * <p>
57443  
57444  * The config options listed for this class are options which may appear in each
57445  * individual column definition.
57446  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57447  * @constructor
57448  * @param {Object} config An Array of column config objects. See this class's
57449  * config objects for details.
57450 */
57451 Roo.grid.ColumnModel = function(config){
57452         /**
57453      * The config passed into the constructor
57454      */
57455     this.config = config;
57456     this.lookup = {};
57457
57458     // if no id, create one
57459     // if the column does not have a dataIndex mapping,
57460     // map it to the order it is in the config
57461     for(var i = 0, len = config.length; i < len; i++){
57462         var c = config[i];
57463         if(typeof c.dataIndex == "undefined"){
57464             c.dataIndex = i;
57465         }
57466         if(typeof c.renderer == "string"){
57467             c.renderer = Roo.util.Format[c.renderer];
57468         }
57469         if(typeof c.id == "undefined"){
57470             c.id = Roo.id();
57471         }
57472         if(c.editor && c.editor.xtype){
57473             c.editor  = Roo.factory(c.editor, Roo.grid);
57474         }
57475         if(c.editor && c.editor.isFormField){
57476             c.editor = new Roo.grid.GridEditor(c.editor);
57477         }
57478         this.lookup[c.id] = c;
57479     }
57480
57481     /**
57482      * The width of columns which have no width specified (defaults to 100)
57483      * @type Number
57484      */
57485     this.defaultWidth = 100;
57486
57487     /**
57488      * Default sortable of columns which have no sortable specified (defaults to false)
57489      * @type Boolean
57490      */
57491     this.defaultSortable = false;
57492
57493     this.addEvents({
57494         /**
57495              * @event widthchange
57496              * Fires when the width of a column changes.
57497              * @param {ColumnModel} this
57498              * @param {Number} columnIndex The column index
57499              * @param {Number} newWidth The new width
57500              */
57501             "widthchange": true,
57502         /**
57503              * @event headerchange
57504              * Fires when the text of a header changes.
57505              * @param {ColumnModel} this
57506              * @param {Number} columnIndex The column index
57507              * @param {Number} newText The new header text
57508              */
57509             "headerchange": true,
57510         /**
57511              * @event hiddenchange
57512              * Fires when a column is hidden or "unhidden".
57513              * @param {ColumnModel} this
57514              * @param {Number} columnIndex The column index
57515              * @param {Boolean} hidden true if hidden, false otherwise
57516              */
57517             "hiddenchange": true,
57518             /**
57519          * @event columnmoved
57520          * Fires when a column is moved.
57521          * @param {ColumnModel} this
57522          * @param {Number} oldIndex
57523          * @param {Number} newIndex
57524          */
57525         "columnmoved" : true,
57526         /**
57527          * @event columlockchange
57528          * Fires when a column's locked state is changed
57529          * @param {ColumnModel} this
57530          * @param {Number} colIndex
57531          * @param {Boolean} locked true if locked
57532          */
57533         "columnlockchange" : true
57534     });
57535     Roo.grid.ColumnModel.superclass.constructor.call(this);
57536 };
57537 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57538     /**
57539      * @cfg {String} header The header text to display in the Grid view.
57540      */
57541     /**
57542      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57543      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57544      * specified, the column's index is used as an index into the Record's data Array.
57545      */
57546     /**
57547      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57548      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57549      */
57550     /**
57551      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57552      * Defaults to the value of the {@link #defaultSortable} property.
57553      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57554      */
57555     /**
57556      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57557      */
57558     /**
57559      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57560      */
57561     /**
57562      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57563      */
57564     /**
57565      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57566      */
57567     /**
57568      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57569      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57570      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57571      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57572      */
57573        /**
57574      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57575      */
57576     /**
57577      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57578      */
57579     /**
57580      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57581      */
57582     /**
57583      * @cfg {String} cursor (Optional)
57584      */
57585     /**
57586      * @cfg {String} tooltip (Optional)
57587      */
57588     /**
57589      * @cfg {Number} xs (Optional)
57590      */
57591     /**
57592      * @cfg {Number} sm (Optional)
57593      */
57594     /**
57595      * @cfg {Number} md (Optional)
57596      */
57597     /**
57598      * @cfg {Number} lg (Optional)
57599      */
57600     /**
57601      * Returns the id of the column at the specified index.
57602      * @param {Number} index The column index
57603      * @return {String} the id
57604      */
57605     getColumnId : function(index){
57606         return this.config[index].id;
57607     },
57608
57609     /**
57610      * Returns the column for a specified id.
57611      * @param {String} id The column id
57612      * @return {Object} the column
57613      */
57614     getColumnById : function(id){
57615         return this.lookup[id];
57616     },
57617
57618     
57619     /**
57620      * Returns the column for a specified dataIndex.
57621      * @param {String} dataIndex The column dataIndex
57622      * @return {Object|Boolean} the column or false if not found
57623      */
57624     getColumnByDataIndex: function(dataIndex){
57625         var index = this.findColumnIndex(dataIndex);
57626         return index > -1 ? this.config[index] : false;
57627     },
57628     
57629     /**
57630      * Returns the index for a specified column id.
57631      * @param {String} id The column id
57632      * @return {Number} the index, or -1 if not found
57633      */
57634     getIndexById : function(id){
57635         for(var i = 0, len = this.config.length; i < len; i++){
57636             if(this.config[i].id == id){
57637                 return i;
57638             }
57639         }
57640         return -1;
57641     },
57642     
57643     /**
57644      * Returns the index for a specified column dataIndex.
57645      * @param {String} dataIndex The column dataIndex
57646      * @return {Number} the index, or -1 if not found
57647      */
57648     
57649     findColumnIndex : function(dataIndex){
57650         for(var i = 0, len = this.config.length; i < len; i++){
57651             if(this.config[i].dataIndex == dataIndex){
57652                 return i;
57653             }
57654         }
57655         return -1;
57656     },
57657     
57658     
57659     moveColumn : function(oldIndex, newIndex){
57660         var c = this.config[oldIndex];
57661         this.config.splice(oldIndex, 1);
57662         this.config.splice(newIndex, 0, c);
57663         this.dataMap = null;
57664         this.fireEvent("columnmoved", this, oldIndex, newIndex);
57665     },
57666
57667     isLocked : function(colIndex){
57668         return this.config[colIndex].locked === true;
57669     },
57670
57671     setLocked : function(colIndex, value, suppressEvent){
57672         if(this.isLocked(colIndex) == value){
57673             return;
57674         }
57675         this.config[colIndex].locked = value;
57676         if(!suppressEvent){
57677             this.fireEvent("columnlockchange", this, colIndex, value);
57678         }
57679     },
57680
57681     getTotalLockedWidth : function(){
57682         var totalWidth = 0;
57683         for(var i = 0; i < this.config.length; i++){
57684             if(this.isLocked(i) && !this.isHidden(i)){
57685                 this.totalWidth += this.getColumnWidth(i);
57686             }
57687         }
57688         return totalWidth;
57689     },
57690
57691     getLockedCount : function(){
57692         for(var i = 0, len = this.config.length; i < len; i++){
57693             if(!this.isLocked(i)){
57694                 return i;
57695             }
57696         }
57697         
57698         return this.config.length;
57699     },
57700
57701     /**
57702      * Returns the number of columns.
57703      * @return {Number}
57704      */
57705     getColumnCount : function(visibleOnly){
57706         if(visibleOnly === true){
57707             var c = 0;
57708             for(var i = 0, len = this.config.length; i < len; i++){
57709                 if(!this.isHidden(i)){
57710                     c++;
57711                 }
57712             }
57713             return c;
57714         }
57715         return this.config.length;
57716     },
57717
57718     /**
57719      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
57720      * @param {Function} fn
57721      * @param {Object} scope (optional)
57722      * @return {Array} result
57723      */
57724     getColumnsBy : function(fn, scope){
57725         var r = [];
57726         for(var i = 0, len = this.config.length; i < len; i++){
57727             var c = this.config[i];
57728             if(fn.call(scope||this, c, i) === true){
57729                 r[r.length] = c;
57730             }
57731         }
57732         return r;
57733     },
57734
57735     /**
57736      * Returns true if the specified column is sortable.
57737      * @param {Number} col The column index
57738      * @return {Boolean}
57739      */
57740     isSortable : function(col){
57741         if(typeof this.config[col].sortable == "undefined"){
57742             return this.defaultSortable;
57743         }
57744         return this.config[col].sortable;
57745     },
57746
57747     /**
57748      * Returns the rendering (formatting) function defined for the column.
57749      * @param {Number} col The column index.
57750      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
57751      */
57752     getRenderer : function(col){
57753         if(!this.config[col].renderer){
57754             return Roo.grid.ColumnModel.defaultRenderer;
57755         }
57756         return this.config[col].renderer;
57757     },
57758
57759     /**
57760      * Sets the rendering (formatting) function for a column.
57761      * @param {Number} col The column index
57762      * @param {Function} fn The function to use to process the cell's raw data
57763      * to return HTML markup for the grid view. The render function is called with
57764      * the following parameters:<ul>
57765      * <li>Data value.</li>
57766      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
57767      * <li>css A CSS style string to apply to the table cell.</li>
57768      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
57769      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
57770      * <li>Row index</li>
57771      * <li>Column index</li>
57772      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
57773      */
57774     setRenderer : function(col, fn){
57775         this.config[col].renderer = fn;
57776     },
57777
57778     /**
57779      * Returns the width for the specified column.
57780      * @param {Number} col The column index
57781      * @return {Number}
57782      */
57783     getColumnWidth : function(col){
57784         return this.config[col].width * 1 || this.defaultWidth;
57785     },
57786
57787     /**
57788      * Sets the width for a column.
57789      * @param {Number} col The column index
57790      * @param {Number} width The new width
57791      */
57792     setColumnWidth : function(col, width, suppressEvent){
57793         this.config[col].width = width;
57794         this.totalWidth = null;
57795         if(!suppressEvent){
57796              this.fireEvent("widthchange", this, col, width);
57797         }
57798     },
57799
57800     /**
57801      * Returns the total width of all columns.
57802      * @param {Boolean} includeHidden True to include hidden column widths
57803      * @return {Number}
57804      */
57805     getTotalWidth : function(includeHidden){
57806         if(!this.totalWidth){
57807             this.totalWidth = 0;
57808             for(var i = 0, len = this.config.length; i < len; i++){
57809                 if(includeHidden || !this.isHidden(i)){
57810                     this.totalWidth += this.getColumnWidth(i);
57811                 }
57812             }
57813         }
57814         return this.totalWidth;
57815     },
57816
57817     /**
57818      * Returns the header for the specified column.
57819      * @param {Number} col The column index
57820      * @return {String}
57821      */
57822     getColumnHeader : function(col){
57823         return this.config[col].header;
57824     },
57825
57826     /**
57827      * Sets the header for a column.
57828      * @param {Number} col The column index
57829      * @param {String} header The new header
57830      */
57831     setColumnHeader : function(col, header){
57832         this.config[col].header = header;
57833         this.fireEvent("headerchange", this, col, header);
57834     },
57835
57836     /**
57837      * Returns the tooltip for the specified column.
57838      * @param {Number} col The column index
57839      * @return {String}
57840      */
57841     getColumnTooltip : function(col){
57842             return this.config[col].tooltip;
57843     },
57844     /**
57845      * Sets the tooltip for a column.
57846      * @param {Number} col The column index
57847      * @param {String} tooltip The new tooltip
57848      */
57849     setColumnTooltip : function(col, tooltip){
57850             this.config[col].tooltip = tooltip;
57851     },
57852
57853     /**
57854      * Returns the dataIndex for the specified column.
57855      * @param {Number} col The column index
57856      * @return {Number}
57857      */
57858     getDataIndex : function(col){
57859         return this.config[col].dataIndex;
57860     },
57861
57862     /**
57863      * Sets the dataIndex for a column.
57864      * @param {Number} col The column index
57865      * @param {Number} dataIndex The new dataIndex
57866      */
57867     setDataIndex : function(col, dataIndex){
57868         this.config[col].dataIndex = dataIndex;
57869     },
57870
57871     
57872     
57873     /**
57874      * Returns true if the cell is editable.
57875      * @param {Number} colIndex The column index
57876      * @param {Number} rowIndex The row index - this is nto actually used..?
57877      * @return {Boolean}
57878      */
57879     isCellEditable : function(colIndex, rowIndex){
57880         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
57881     },
57882
57883     /**
57884      * Returns the editor defined for the cell/column.
57885      * return false or null to disable editing.
57886      * @param {Number} colIndex The column index
57887      * @param {Number} rowIndex The row index
57888      * @return {Object}
57889      */
57890     getCellEditor : function(colIndex, rowIndex){
57891         return this.config[colIndex].editor;
57892     },
57893
57894     /**
57895      * Sets if a column is editable.
57896      * @param {Number} col The column index
57897      * @param {Boolean} editable True if the column is editable
57898      */
57899     setEditable : function(col, editable){
57900         this.config[col].editable = editable;
57901     },
57902
57903
57904     /**
57905      * Returns true if the column is hidden.
57906      * @param {Number} colIndex The column index
57907      * @return {Boolean}
57908      */
57909     isHidden : function(colIndex){
57910         return this.config[colIndex].hidden;
57911     },
57912
57913
57914     /**
57915      * Returns true if the column width cannot be changed
57916      */
57917     isFixed : function(colIndex){
57918         return this.config[colIndex].fixed;
57919     },
57920
57921     /**
57922      * Returns true if the column can be resized
57923      * @return {Boolean}
57924      */
57925     isResizable : function(colIndex){
57926         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
57927     },
57928     /**
57929      * Sets if a column is hidden.
57930      * @param {Number} colIndex The column index
57931      * @param {Boolean} hidden True if the column is hidden
57932      */
57933     setHidden : function(colIndex, hidden){
57934         this.config[colIndex].hidden = hidden;
57935         this.totalWidth = null;
57936         this.fireEvent("hiddenchange", this, colIndex, hidden);
57937     },
57938
57939     /**
57940      * Sets the editor for a column.
57941      * @param {Number} col The column index
57942      * @param {Object} editor The editor object
57943      */
57944     setEditor : function(col, editor){
57945         this.config[col].editor = editor;
57946     }
57947 });
57948
57949 Roo.grid.ColumnModel.defaultRenderer = function(value)
57950 {
57951     if(typeof value == "object") {
57952         return value;
57953     }
57954         if(typeof value == "string" && value.length < 1){
57955             return "&#160;";
57956         }
57957     
57958         return String.format("{0}", value);
57959 };
57960
57961 // Alias for backwards compatibility
57962 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
57963 /*
57964  * Based on:
57965  * Ext JS Library 1.1.1
57966  * Copyright(c) 2006-2007, Ext JS, LLC.
57967  *
57968  * Originally Released Under LGPL - original licence link has changed is not relivant.
57969  *
57970  * Fork - LGPL
57971  * <script type="text/javascript">
57972  */
57973
57974 /**
57975  * @class Roo.grid.AbstractSelectionModel
57976  * @extends Roo.util.Observable
57977  * Abstract base class for grid SelectionModels.  It provides the interface that should be
57978  * implemented by descendant classes.  This class should not be directly instantiated.
57979  * @constructor
57980  */
57981 Roo.grid.AbstractSelectionModel = function(){
57982     this.locked = false;
57983     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
57984 };
57985
57986 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
57987     /** @ignore Called by the grid automatically. Do not call directly. */
57988     init : function(grid){
57989         this.grid = grid;
57990         this.initEvents();
57991     },
57992
57993     /**
57994      * Locks the selections.
57995      */
57996     lock : function(){
57997         this.locked = true;
57998     },
57999
58000     /**
58001      * Unlocks the selections.
58002      */
58003     unlock : function(){
58004         this.locked = false;
58005     },
58006
58007     /**
58008      * Returns true if the selections are locked.
58009      * @return {Boolean}
58010      */
58011     isLocked : function(){
58012         return this.locked;
58013     }
58014 });/*
58015  * Based on:
58016  * Ext JS Library 1.1.1
58017  * Copyright(c) 2006-2007, Ext JS, LLC.
58018  *
58019  * Originally Released Under LGPL - original licence link has changed is not relivant.
58020  *
58021  * Fork - LGPL
58022  * <script type="text/javascript">
58023  */
58024 /**
58025  * @extends Roo.grid.AbstractSelectionModel
58026  * @class Roo.grid.RowSelectionModel
58027  * The default SelectionModel used by {@link Roo.grid.Grid}.
58028  * It supports multiple selections and keyboard selection/navigation. 
58029  * @constructor
58030  * @param {Object} config
58031  */
58032 Roo.grid.RowSelectionModel = function(config){
58033     Roo.apply(this, config);
58034     this.selections = new Roo.util.MixedCollection(false, function(o){
58035         return o.id;
58036     });
58037
58038     this.last = false;
58039     this.lastActive = false;
58040
58041     this.addEvents({
58042         /**
58043              * @event selectionchange
58044              * Fires when the selection changes
58045              * @param {SelectionModel} this
58046              */
58047             "selectionchange" : true,
58048         /**
58049              * @event afterselectionchange
58050              * Fires after the selection changes (eg. by key press or clicking)
58051              * @param {SelectionModel} this
58052              */
58053             "afterselectionchange" : true,
58054         /**
58055              * @event beforerowselect
58056              * Fires when a row is selected being selected, return false to cancel.
58057              * @param {SelectionModel} this
58058              * @param {Number} rowIndex The selected index
58059              * @param {Boolean} keepExisting False if other selections will be cleared
58060              */
58061             "beforerowselect" : true,
58062         /**
58063              * @event rowselect
58064              * Fires when a row is selected.
58065              * @param {SelectionModel} this
58066              * @param {Number} rowIndex The selected index
58067              * @param {Roo.data.Record} r The record
58068              */
58069             "rowselect" : true,
58070         /**
58071              * @event rowdeselect
58072              * Fires when a row is deselected.
58073              * @param {SelectionModel} this
58074              * @param {Number} rowIndex The selected index
58075              */
58076         "rowdeselect" : true
58077     });
58078     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58079     this.locked = false;
58080 };
58081
58082 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58083     /**
58084      * @cfg {Boolean} singleSelect
58085      * True to allow selection of only one row at a time (defaults to false)
58086      */
58087     singleSelect : false,
58088
58089     // private
58090     initEvents : function(){
58091
58092         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58093             this.grid.on("mousedown", this.handleMouseDown, this);
58094         }else{ // allow click to work like normal
58095             this.grid.on("rowclick", this.handleDragableRowClick, this);
58096         }
58097
58098         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58099             "up" : function(e){
58100                 if(!e.shiftKey){
58101                     this.selectPrevious(e.shiftKey);
58102                 }else if(this.last !== false && this.lastActive !== false){
58103                     var last = this.last;
58104                     this.selectRange(this.last,  this.lastActive-1);
58105                     this.grid.getView().focusRow(this.lastActive);
58106                     if(last !== false){
58107                         this.last = last;
58108                     }
58109                 }else{
58110                     this.selectFirstRow();
58111                 }
58112                 this.fireEvent("afterselectionchange", this);
58113             },
58114             "down" : function(e){
58115                 if(!e.shiftKey){
58116                     this.selectNext(e.shiftKey);
58117                 }else if(this.last !== false && this.lastActive !== false){
58118                     var last = this.last;
58119                     this.selectRange(this.last,  this.lastActive+1);
58120                     this.grid.getView().focusRow(this.lastActive);
58121                     if(last !== false){
58122                         this.last = last;
58123                     }
58124                 }else{
58125                     this.selectFirstRow();
58126                 }
58127                 this.fireEvent("afterselectionchange", this);
58128             },
58129             scope: this
58130         });
58131
58132         var view = this.grid.view;
58133         view.on("refresh", this.onRefresh, this);
58134         view.on("rowupdated", this.onRowUpdated, this);
58135         view.on("rowremoved", this.onRemove, this);
58136     },
58137
58138     // private
58139     onRefresh : function(){
58140         var ds = this.grid.dataSource, i, v = this.grid.view;
58141         var s = this.selections;
58142         s.each(function(r){
58143             if((i = ds.indexOfId(r.id)) != -1){
58144                 v.onRowSelect(i);
58145                 s.add(ds.getAt(i)); // updating the selection relate data
58146             }else{
58147                 s.remove(r);
58148             }
58149         });
58150     },
58151
58152     // private
58153     onRemove : function(v, index, r){
58154         this.selections.remove(r);
58155     },
58156
58157     // private
58158     onRowUpdated : function(v, index, r){
58159         if(this.isSelected(r)){
58160             v.onRowSelect(index);
58161         }
58162     },
58163
58164     /**
58165      * Select records.
58166      * @param {Array} records The records to select
58167      * @param {Boolean} keepExisting (optional) True to keep existing selections
58168      */
58169     selectRecords : function(records, keepExisting){
58170         if(!keepExisting){
58171             this.clearSelections();
58172         }
58173         var ds = this.grid.dataSource;
58174         for(var i = 0, len = records.length; i < len; i++){
58175             this.selectRow(ds.indexOf(records[i]), true);
58176         }
58177     },
58178
58179     /**
58180      * Gets the number of selected rows.
58181      * @return {Number}
58182      */
58183     getCount : function(){
58184         return this.selections.length;
58185     },
58186
58187     /**
58188      * Selects the first row in the grid.
58189      */
58190     selectFirstRow : function(){
58191         this.selectRow(0);
58192     },
58193
58194     /**
58195      * Select the last row.
58196      * @param {Boolean} keepExisting (optional) True to keep existing selections
58197      */
58198     selectLastRow : function(keepExisting){
58199         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58200     },
58201
58202     /**
58203      * Selects the row immediately following the last selected row.
58204      * @param {Boolean} keepExisting (optional) True to keep existing selections
58205      */
58206     selectNext : function(keepExisting){
58207         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58208             this.selectRow(this.last+1, keepExisting);
58209             this.grid.getView().focusRow(this.last);
58210         }
58211     },
58212
58213     /**
58214      * Selects the row that precedes the last selected row.
58215      * @param {Boolean} keepExisting (optional) True to keep existing selections
58216      */
58217     selectPrevious : function(keepExisting){
58218         if(this.last){
58219             this.selectRow(this.last-1, keepExisting);
58220             this.grid.getView().focusRow(this.last);
58221         }
58222     },
58223
58224     /**
58225      * Returns the selected records
58226      * @return {Array} Array of selected records
58227      */
58228     getSelections : function(){
58229         return [].concat(this.selections.items);
58230     },
58231
58232     /**
58233      * Returns the first selected record.
58234      * @return {Record}
58235      */
58236     getSelected : function(){
58237         return this.selections.itemAt(0);
58238     },
58239
58240
58241     /**
58242      * Clears all selections.
58243      */
58244     clearSelections : function(fast){
58245         if(this.locked) {
58246             return;
58247         }
58248         if(fast !== true){
58249             var ds = this.grid.dataSource;
58250             var s = this.selections;
58251             s.each(function(r){
58252                 this.deselectRow(ds.indexOfId(r.id));
58253             }, this);
58254             s.clear();
58255         }else{
58256             this.selections.clear();
58257         }
58258         this.last = false;
58259     },
58260
58261
58262     /**
58263      * Selects all rows.
58264      */
58265     selectAll : function(){
58266         if(this.locked) {
58267             return;
58268         }
58269         this.selections.clear();
58270         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58271             this.selectRow(i, true);
58272         }
58273     },
58274
58275     /**
58276      * Returns True if there is a selection.
58277      * @return {Boolean}
58278      */
58279     hasSelection : function(){
58280         return this.selections.length > 0;
58281     },
58282
58283     /**
58284      * Returns True if the specified row is selected.
58285      * @param {Number/Record} record The record or index of the record to check
58286      * @return {Boolean}
58287      */
58288     isSelected : function(index){
58289         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58290         return (r && this.selections.key(r.id) ? true : false);
58291     },
58292
58293     /**
58294      * Returns True if the specified record id is selected.
58295      * @param {String} id The id of record to check
58296      * @return {Boolean}
58297      */
58298     isIdSelected : function(id){
58299         return (this.selections.key(id) ? true : false);
58300     },
58301
58302     // private
58303     handleMouseDown : function(e, t){
58304         var view = this.grid.getView(), rowIndex;
58305         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58306             return;
58307         };
58308         if(e.shiftKey && this.last !== false){
58309             var last = this.last;
58310             this.selectRange(last, rowIndex, e.ctrlKey);
58311             this.last = last; // reset the last
58312             view.focusRow(rowIndex);
58313         }else{
58314             var isSelected = this.isSelected(rowIndex);
58315             if(e.button !== 0 && isSelected){
58316                 view.focusRow(rowIndex);
58317             }else if(e.ctrlKey && isSelected){
58318                 this.deselectRow(rowIndex);
58319             }else if(!isSelected){
58320                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58321                 view.focusRow(rowIndex);
58322             }
58323         }
58324         this.fireEvent("afterselectionchange", this);
58325     },
58326     // private
58327     handleDragableRowClick :  function(grid, rowIndex, e) 
58328     {
58329         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58330             this.selectRow(rowIndex, false);
58331             grid.view.focusRow(rowIndex);
58332              this.fireEvent("afterselectionchange", this);
58333         }
58334     },
58335     
58336     /**
58337      * Selects multiple rows.
58338      * @param {Array} rows Array of the indexes of the row to select
58339      * @param {Boolean} keepExisting (optional) True to keep existing selections
58340      */
58341     selectRows : function(rows, keepExisting){
58342         if(!keepExisting){
58343             this.clearSelections();
58344         }
58345         for(var i = 0, len = rows.length; i < len; i++){
58346             this.selectRow(rows[i], true);
58347         }
58348     },
58349
58350     /**
58351      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58352      * @param {Number} startRow The index of the first row in the range
58353      * @param {Number} endRow The index of the last row in the range
58354      * @param {Boolean} keepExisting (optional) True to retain existing selections
58355      */
58356     selectRange : function(startRow, endRow, keepExisting){
58357         if(this.locked) {
58358             return;
58359         }
58360         if(!keepExisting){
58361             this.clearSelections();
58362         }
58363         if(startRow <= endRow){
58364             for(var i = startRow; i <= endRow; i++){
58365                 this.selectRow(i, true);
58366             }
58367         }else{
58368             for(var i = startRow; i >= endRow; i--){
58369                 this.selectRow(i, true);
58370             }
58371         }
58372     },
58373
58374     /**
58375      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58376      * @param {Number} startRow The index of the first row in the range
58377      * @param {Number} endRow The index of the last row in the range
58378      */
58379     deselectRange : function(startRow, endRow, preventViewNotify){
58380         if(this.locked) {
58381             return;
58382         }
58383         for(var i = startRow; i <= endRow; i++){
58384             this.deselectRow(i, preventViewNotify);
58385         }
58386     },
58387
58388     /**
58389      * Selects a row.
58390      * @param {Number} row The index of the row to select
58391      * @param {Boolean} keepExisting (optional) True to keep existing selections
58392      */
58393     selectRow : function(index, keepExisting, preventViewNotify){
58394         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58395             return;
58396         }
58397         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58398             if(!keepExisting || this.singleSelect){
58399                 this.clearSelections();
58400             }
58401             var r = this.grid.dataSource.getAt(index);
58402             this.selections.add(r);
58403             this.last = this.lastActive = index;
58404             if(!preventViewNotify){
58405                 this.grid.getView().onRowSelect(index);
58406             }
58407             this.fireEvent("rowselect", this, index, r);
58408             this.fireEvent("selectionchange", this);
58409         }
58410     },
58411
58412     /**
58413      * Deselects a row.
58414      * @param {Number} row The index of the row to deselect
58415      */
58416     deselectRow : function(index, preventViewNotify){
58417         if(this.locked) {
58418             return;
58419         }
58420         if(this.last == index){
58421             this.last = false;
58422         }
58423         if(this.lastActive == index){
58424             this.lastActive = false;
58425         }
58426         var r = this.grid.dataSource.getAt(index);
58427         this.selections.remove(r);
58428         if(!preventViewNotify){
58429             this.grid.getView().onRowDeselect(index);
58430         }
58431         this.fireEvent("rowdeselect", this, index);
58432         this.fireEvent("selectionchange", this);
58433     },
58434
58435     // private
58436     restoreLast : function(){
58437         if(this._last){
58438             this.last = this._last;
58439         }
58440     },
58441
58442     // private
58443     acceptsNav : function(row, col, cm){
58444         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58445     },
58446
58447     // private
58448     onEditorKey : function(field, e){
58449         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58450         if(k == e.TAB){
58451             e.stopEvent();
58452             ed.completeEdit();
58453             if(e.shiftKey){
58454                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58455             }else{
58456                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58457             }
58458         }else if(k == e.ENTER && !e.ctrlKey){
58459             e.stopEvent();
58460             ed.completeEdit();
58461             if(e.shiftKey){
58462                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58463             }else{
58464                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58465             }
58466         }else if(k == e.ESC){
58467             ed.cancelEdit();
58468         }
58469         if(newCell){
58470             g.startEditing(newCell[0], newCell[1]);
58471         }
58472     }
58473 });/*
58474  * Based on:
58475  * Ext JS Library 1.1.1
58476  * Copyright(c) 2006-2007, Ext JS, LLC.
58477  *
58478  * Originally Released Under LGPL - original licence link has changed is not relivant.
58479  *
58480  * Fork - LGPL
58481  * <script type="text/javascript">
58482  */
58483 /**
58484  * @class Roo.grid.CellSelectionModel
58485  * @extends Roo.grid.AbstractSelectionModel
58486  * This class provides the basic implementation for cell selection in a grid.
58487  * @constructor
58488  * @param {Object} config The object containing the configuration of this model.
58489  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58490  */
58491 Roo.grid.CellSelectionModel = function(config){
58492     Roo.apply(this, config);
58493
58494     this.selection = null;
58495
58496     this.addEvents({
58497         /**
58498              * @event beforerowselect
58499              * Fires before a cell is selected.
58500              * @param {SelectionModel} this
58501              * @param {Number} rowIndex The selected row index
58502              * @param {Number} colIndex The selected cell index
58503              */
58504             "beforecellselect" : true,
58505         /**
58506              * @event cellselect
58507              * Fires when a cell is selected.
58508              * @param {SelectionModel} this
58509              * @param {Number} rowIndex The selected row index
58510              * @param {Number} colIndex The selected cell index
58511              */
58512             "cellselect" : true,
58513         /**
58514              * @event selectionchange
58515              * Fires when the active selection changes.
58516              * @param {SelectionModel} this
58517              * @param {Object} selection null for no selection or an object (o) with two properties
58518                 <ul>
58519                 <li>o.record: the record object for the row the selection is in</li>
58520                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58521                 </ul>
58522              */
58523             "selectionchange" : true,
58524         /**
58525              * @event tabend
58526              * Fires when the tab (or enter) was pressed on the last editable cell
58527              * You can use this to trigger add new row.
58528              * @param {SelectionModel} this
58529              */
58530             "tabend" : true,
58531          /**
58532              * @event beforeeditnext
58533              * Fires before the next editable sell is made active
58534              * You can use this to skip to another cell or fire the tabend
58535              *    if you set cell to false
58536              * @param {Object} eventdata object : { cell : [ row, col ] } 
58537              */
58538             "beforeeditnext" : true
58539     });
58540     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58541 };
58542
58543 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58544     
58545     enter_is_tab: false,
58546
58547     /** @ignore */
58548     initEvents : function(){
58549         this.grid.on("mousedown", this.handleMouseDown, this);
58550         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58551         var view = this.grid.view;
58552         view.on("refresh", this.onViewChange, this);
58553         view.on("rowupdated", this.onRowUpdated, this);
58554         view.on("beforerowremoved", this.clearSelections, this);
58555         view.on("beforerowsinserted", this.clearSelections, this);
58556         if(this.grid.isEditor){
58557             this.grid.on("beforeedit", this.beforeEdit,  this);
58558         }
58559     },
58560
58561         //private
58562     beforeEdit : function(e){
58563         this.select(e.row, e.column, false, true, e.record);
58564     },
58565
58566         //private
58567     onRowUpdated : function(v, index, r){
58568         if(this.selection && this.selection.record == r){
58569             v.onCellSelect(index, this.selection.cell[1]);
58570         }
58571     },
58572
58573         //private
58574     onViewChange : function(){
58575         this.clearSelections(true);
58576     },
58577
58578         /**
58579          * Returns the currently selected cell,.
58580          * @return {Array} The selected cell (row, column) or null if none selected.
58581          */
58582     getSelectedCell : function(){
58583         return this.selection ? this.selection.cell : null;
58584     },
58585
58586     /**
58587      * Clears all selections.
58588      * @param {Boolean} true to prevent the gridview from being notified about the change.
58589      */
58590     clearSelections : function(preventNotify){
58591         var s = this.selection;
58592         if(s){
58593             if(preventNotify !== true){
58594                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
58595             }
58596             this.selection = null;
58597             this.fireEvent("selectionchange", this, null);
58598         }
58599     },
58600
58601     /**
58602      * Returns true if there is a selection.
58603      * @return {Boolean}
58604      */
58605     hasSelection : function(){
58606         return this.selection ? true : false;
58607     },
58608
58609     /** @ignore */
58610     handleMouseDown : function(e, t){
58611         var v = this.grid.getView();
58612         if(this.isLocked()){
58613             return;
58614         };
58615         var row = v.findRowIndex(t);
58616         var cell = v.findCellIndex(t);
58617         if(row !== false && cell !== false){
58618             this.select(row, cell);
58619         }
58620     },
58621
58622     /**
58623      * Selects a cell.
58624      * @param {Number} rowIndex
58625      * @param {Number} collIndex
58626      */
58627     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
58628         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
58629             this.clearSelections();
58630             r = r || this.grid.dataSource.getAt(rowIndex);
58631             this.selection = {
58632                 record : r,
58633                 cell : [rowIndex, colIndex]
58634             };
58635             if(!preventViewNotify){
58636                 var v = this.grid.getView();
58637                 v.onCellSelect(rowIndex, colIndex);
58638                 if(preventFocus !== true){
58639                     v.focusCell(rowIndex, colIndex);
58640                 }
58641             }
58642             this.fireEvent("cellselect", this, rowIndex, colIndex);
58643             this.fireEvent("selectionchange", this, this.selection);
58644         }
58645     },
58646
58647         //private
58648     isSelectable : function(rowIndex, colIndex, cm){
58649         return !cm.isHidden(colIndex);
58650     },
58651
58652     /** @ignore */
58653     handleKeyDown : function(e){
58654         //Roo.log('Cell Sel Model handleKeyDown');
58655         if(!e.isNavKeyPress()){
58656             return;
58657         }
58658         var g = this.grid, s = this.selection;
58659         if(!s){
58660             e.stopEvent();
58661             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
58662             if(cell){
58663                 this.select(cell[0], cell[1]);
58664             }
58665             return;
58666         }
58667         var sm = this;
58668         var walk = function(row, col, step){
58669             return g.walkCells(row, col, step, sm.isSelectable,  sm);
58670         };
58671         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
58672         var newCell;
58673
58674       
58675
58676         switch(k){
58677             case e.TAB:
58678                 // handled by onEditorKey
58679                 if (g.isEditor && g.editing) {
58680                     return;
58681                 }
58682                 if(e.shiftKey) {
58683                     newCell = walk(r, c-1, -1);
58684                 } else {
58685                     newCell = walk(r, c+1, 1);
58686                 }
58687                 break;
58688             
58689             case e.DOWN:
58690                newCell = walk(r+1, c, 1);
58691                 break;
58692             
58693             case e.UP:
58694                 newCell = walk(r-1, c, -1);
58695                 break;
58696             
58697             case e.RIGHT:
58698                 newCell = walk(r, c+1, 1);
58699                 break;
58700             
58701             case e.LEFT:
58702                 newCell = walk(r, c-1, -1);
58703                 break;
58704             
58705             case e.ENTER:
58706                 
58707                 if(g.isEditor && !g.editing){
58708                    g.startEditing(r, c);
58709                    e.stopEvent();
58710                    return;
58711                 }
58712                 
58713                 
58714              break;
58715         };
58716         if(newCell){
58717             this.select(newCell[0], newCell[1]);
58718             e.stopEvent();
58719             
58720         }
58721     },
58722
58723     acceptsNav : function(row, col, cm){
58724         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58725     },
58726     /**
58727      * Selects a cell.
58728      * @param {Number} field (not used) - as it's normally used as a listener
58729      * @param {Number} e - event - fake it by using
58730      *
58731      * var e = Roo.EventObjectImpl.prototype;
58732      * e.keyCode = e.TAB
58733      *
58734      * 
58735      */
58736     onEditorKey : function(field, e){
58737         
58738         var k = e.getKey(),
58739             newCell,
58740             g = this.grid,
58741             ed = g.activeEditor,
58742             forward = false;
58743         ///Roo.log('onEditorKey' + k);
58744         
58745         
58746         if (this.enter_is_tab && k == e.ENTER) {
58747             k = e.TAB;
58748         }
58749         
58750         if(k == e.TAB){
58751             if(e.shiftKey){
58752                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58753             }else{
58754                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58755                 forward = true;
58756             }
58757             
58758             e.stopEvent();
58759             
58760         } else if(k == e.ENTER &&  !e.ctrlKey){
58761             ed.completeEdit();
58762             e.stopEvent();
58763             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58764         
58765                 } else if(k == e.ESC){
58766             ed.cancelEdit();
58767         }
58768                 
58769         if (newCell) {
58770             var ecall = { cell : newCell, forward : forward };
58771             this.fireEvent('beforeeditnext', ecall );
58772             newCell = ecall.cell;
58773                         forward = ecall.forward;
58774         }
58775                 
58776         if(newCell){
58777             //Roo.log('next cell after edit');
58778             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
58779         } else if (forward) {
58780             // tabbed past last
58781             this.fireEvent.defer(100, this, ['tabend',this]);
58782         }
58783     }
58784 });/*
58785  * Based on:
58786  * Ext JS Library 1.1.1
58787  * Copyright(c) 2006-2007, Ext JS, LLC.
58788  *
58789  * Originally Released Under LGPL - original licence link has changed is not relivant.
58790  *
58791  * Fork - LGPL
58792  * <script type="text/javascript">
58793  */
58794  
58795 /**
58796  * @class Roo.grid.EditorGrid
58797  * @extends Roo.grid.Grid
58798  * Class for creating and editable grid.
58799  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
58800  * The container MUST have some type of size defined for the grid to fill. The container will be 
58801  * automatically set to position relative if it isn't already.
58802  * @param {Object} dataSource The data model to bind to
58803  * @param {Object} colModel The column model with info about this grid's columns
58804  */
58805 Roo.grid.EditorGrid = function(container, config){
58806     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
58807     this.getGridEl().addClass("xedit-grid");
58808
58809     if(!this.selModel){
58810         this.selModel = new Roo.grid.CellSelectionModel();
58811     }
58812
58813     this.activeEditor = null;
58814
58815         this.addEvents({
58816             /**
58817              * @event beforeedit
58818              * Fires before cell editing is triggered. The edit event object has the following properties <br />
58819              * <ul style="padding:5px;padding-left:16px;">
58820              * <li>grid - This grid</li>
58821              * <li>record - The record being edited</li>
58822              * <li>field - The field name being edited</li>
58823              * <li>value - The value for the field being edited.</li>
58824              * <li>row - The grid row index</li>
58825              * <li>column - The grid column index</li>
58826              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58827              * </ul>
58828              * @param {Object} e An edit event (see above for description)
58829              */
58830             "beforeedit" : true,
58831             /**
58832              * @event afteredit
58833              * Fires after a cell is edited. <br />
58834              * <ul style="padding:5px;padding-left:16px;">
58835              * <li>grid - This grid</li>
58836              * <li>record - The record being edited</li>
58837              * <li>field - The field name being edited</li>
58838              * <li>value - The value being set</li>
58839              * <li>originalValue - The original value for the field, before the edit.</li>
58840              * <li>row - The grid row index</li>
58841              * <li>column - The grid column index</li>
58842              * </ul>
58843              * @param {Object} e An edit event (see above for description)
58844              */
58845             "afteredit" : true,
58846             /**
58847              * @event validateedit
58848              * Fires after a cell is edited, but before the value is set in the record. 
58849          * You can use this to modify the value being set in the field, Return false
58850              * to cancel the change. The edit event object has the following properties <br />
58851              * <ul style="padding:5px;padding-left:16px;">
58852          * <li>editor - This editor</li>
58853              * <li>grid - This grid</li>
58854              * <li>record - The record being edited</li>
58855              * <li>field - The field name being edited</li>
58856              * <li>value - The value being set</li>
58857              * <li>originalValue - The original value for the field, before the edit.</li>
58858              * <li>row - The grid row index</li>
58859              * <li>column - The grid column index</li>
58860              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58861              * </ul>
58862              * @param {Object} e An edit event (see above for description)
58863              */
58864             "validateedit" : true
58865         });
58866     this.on("bodyscroll", this.stopEditing,  this);
58867     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
58868 };
58869
58870 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
58871     /**
58872      * @cfg {Number} clicksToEdit
58873      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
58874      */
58875     clicksToEdit: 2,
58876
58877     // private
58878     isEditor : true,
58879     // private
58880     trackMouseOver: false, // causes very odd FF errors
58881
58882     onCellDblClick : function(g, row, col){
58883         this.startEditing(row, col);
58884     },
58885
58886     onEditComplete : function(ed, value, startValue){
58887         this.editing = false;
58888         this.activeEditor = null;
58889         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
58890         var r = ed.record;
58891         var field = this.colModel.getDataIndex(ed.col);
58892         var e = {
58893             grid: this,
58894             record: r,
58895             field: field,
58896             originalValue: startValue,
58897             value: value,
58898             row: ed.row,
58899             column: ed.col,
58900             cancel:false,
58901             editor: ed
58902         };
58903         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
58904         cell.show();
58905           
58906         if(String(value) !== String(startValue)){
58907             
58908             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
58909                 r.set(field, e.value);
58910                 // if we are dealing with a combo box..
58911                 // then we also set the 'name' colum to be the displayField
58912                 if (ed.field.displayField && ed.field.name) {
58913                     r.set(ed.field.name, ed.field.el.dom.value);
58914                 }
58915                 
58916                 delete e.cancel; //?? why!!!
58917                 this.fireEvent("afteredit", e);
58918             }
58919         } else {
58920             this.fireEvent("afteredit", e); // always fire it!
58921         }
58922         this.view.focusCell(ed.row, ed.col);
58923     },
58924
58925     /**
58926      * Starts editing the specified for the specified row/column
58927      * @param {Number} rowIndex
58928      * @param {Number} colIndex
58929      */
58930     startEditing : function(row, col){
58931         this.stopEditing();
58932         if(this.colModel.isCellEditable(col, row)){
58933             this.view.ensureVisible(row, col, true);
58934           
58935             var r = this.dataSource.getAt(row);
58936             var field = this.colModel.getDataIndex(col);
58937             var cell = Roo.get(this.view.getCell(row,col));
58938             var e = {
58939                 grid: this,
58940                 record: r,
58941                 field: field,
58942                 value: r.data[field],
58943                 row: row,
58944                 column: col,
58945                 cancel:false 
58946             };
58947             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
58948                 this.editing = true;
58949                 var ed = this.colModel.getCellEditor(col, row);
58950                 
58951                 if (!ed) {
58952                     return;
58953                 }
58954                 if(!ed.rendered){
58955                     ed.render(ed.parentEl || document.body);
58956                 }
58957                 ed.field.reset();
58958                
58959                 cell.hide();
58960                 
58961                 (function(){ // complex but required for focus issues in safari, ie and opera
58962                     ed.row = row;
58963                     ed.col = col;
58964                     ed.record = r;
58965                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
58966                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
58967                     this.activeEditor = ed;
58968                     var v = r.data[field];
58969                     ed.startEdit(this.view.getCell(row, col), v);
58970                     // combo's with 'displayField and name set
58971                     if (ed.field.displayField && ed.field.name) {
58972                         ed.field.el.dom.value = r.data[ed.field.name];
58973                     }
58974                     
58975                     
58976                 }).defer(50, this);
58977             }
58978         }
58979     },
58980         
58981     /**
58982      * Stops any active editing
58983      */
58984     stopEditing : function(){
58985         if(this.activeEditor){
58986             this.activeEditor.completeEdit();
58987         }
58988         this.activeEditor = null;
58989     },
58990         
58991          /**
58992      * Called to get grid's drag proxy text, by default returns this.ddText.
58993      * @return {String}
58994      */
58995     getDragDropText : function(){
58996         var count = this.selModel.getSelectedCell() ? 1 : 0;
58997         return String.format(this.ddText, count, count == 1 ? '' : 's');
58998     }
58999         
59000 });/*
59001  * Based on:
59002  * Ext JS Library 1.1.1
59003  * Copyright(c) 2006-2007, Ext JS, LLC.
59004  *
59005  * Originally Released Under LGPL - original licence link has changed is not relivant.
59006  *
59007  * Fork - LGPL
59008  * <script type="text/javascript">
59009  */
59010
59011 // private - not really -- you end up using it !
59012 // This is a support class used internally by the Grid components
59013
59014 /**
59015  * @class Roo.grid.GridEditor
59016  * @extends Roo.Editor
59017  * Class for creating and editable grid elements.
59018  * @param {Object} config any settings (must include field)
59019  */
59020 Roo.grid.GridEditor = function(field, config){
59021     if (!config && field.field) {
59022         config = field;
59023         field = Roo.factory(config.field, Roo.form);
59024     }
59025     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59026     field.monitorTab = false;
59027 };
59028
59029 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59030     
59031     /**
59032      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59033      */
59034     
59035     alignment: "tl-tl",
59036     autoSize: "width",
59037     hideEl : false,
59038     cls: "x-small-editor x-grid-editor",
59039     shim:false,
59040     shadow:"frame"
59041 });/*
59042  * Based on:
59043  * Ext JS Library 1.1.1
59044  * Copyright(c) 2006-2007, Ext JS, LLC.
59045  *
59046  * Originally Released Under LGPL - original licence link has changed is not relivant.
59047  *
59048  * Fork - LGPL
59049  * <script type="text/javascript">
59050  */
59051   
59052
59053   
59054 Roo.grid.PropertyRecord = Roo.data.Record.create([
59055     {name:'name',type:'string'},  'value'
59056 ]);
59057
59058
59059 Roo.grid.PropertyStore = function(grid, source){
59060     this.grid = grid;
59061     this.store = new Roo.data.Store({
59062         recordType : Roo.grid.PropertyRecord
59063     });
59064     this.store.on('update', this.onUpdate,  this);
59065     if(source){
59066         this.setSource(source);
59067     }
59068     Roo.grid.PropertyStore.superclass.constructor.call(this);
59069 };
59070
59071
59072
59073 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59074     setSource : function(o){
59075         this.source = o;
59076         this.store.removeAll();
59077         var data = [];
59078         for(var k in o){
59079             if(this.isEditableValue(o[k])){
59080                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59081             }
59082         }
59083         this.store.loadRecords({records: data}, {}, true);
59084     },
59085
59086     onUpdate : function(ds, record, type){
59087         if(type == Roo.data.Record.EDIT){
59088             var v = record.data['value'];
59089             var oldValue = record.modified['value'];
59090             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59091                 this.source[record.id] = v;
59092                 record.commit();
59093                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59094             }else{
59095                 record.reject();
59096             }
59097         }
59098     },
59099
59100     getProperty : function(row){
59101        return this.store.getAt(row);
59102     },
59103
59104     isEditableValue: function(val){
59105         if(val && val instanceof Date){
59106             return true;
59107         }else if(typeof val == 'object' || typeof val == 'function'){
59108             return false;
59109         }
59110         return true;
59111     },
59112
59113     setValue : function(prop, value){
59114         this.source[prop] = value;
59115         this.store.getById(prop).set('value', value);
59116     },
59117
59118     getSource : function(){
59119         return this.source;
59120     }
59121 });
59122
59123 Roo.grid.PropertyColumnModel = function(grid, store){
59124     this.grid = grid;
59125     var g = Roo.grid;
59126     g.PropertyColumnModel.superclass.constructor.call(this, [
59127         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59128         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59129     ]);
59130     this.store = store;
59131     this.bselect = Roo.DomHelper.append(document.body, {
59132         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59133             {tag: 'option', value: 'true', html: 'true'},
59134             {tag: 'option', value: 'false', html: 'false'}
59135         ]
59136     });
59137     Roo.id(this.bselect);
59138     var f = Roo.form;
59139     this.editors = {
59140         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59141         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59142         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59143         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59144         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59145     };
59146     this.renderCellDelegate = this.renderCell.createDelegate(this);
59147     this.renderPropDelegate = this.renderProp.createDelegate(this);
59148 };
59149
59150 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59151     
59152     
59153     nameText : 'Name',
59154     valueText : 'Value',
59155     
59156     dateFormat : 'm/j/Y',
59157     
59158     
59159     renderDate : function(dateVal){
59160         return dateVal.dateFormat(this.dateFormat);
59161     },
59162
59163     renderBool : function(bVal){
59164         return bVal ? 'true' : 'false';
59165     },
59166
59167     isCellEditable : function(colIndex, rowIndex){
59168         return colIndex == 1;
59169     },
59170
59171     getRenderer : function(col){
59172         return col == 1 ?
59173             this.renderCellDelegate : this.renderPropDelegate;
59174     },
59175
59176     renderProp : function(v){
59177         return this.getPropertyName(v);
59178     },
59179
59180     renderCell : function(val){
59181         var rv = val;
59182         if(val instanceof Date){
59183             rv = this.renderDate(val);
59184         }else if(typeof val == 'boolean'){
59185             rv = this.renderBool(val);
59186         }
59187         return Roo.util.Format.htmlEncode(rv);
59188     },
59189
59190     getPropertyName : function(name){
59191         var pn = this.grid.propertyNames;
59192         return pn && pn[name] ? pn[name] : name;
59193     },
59194
59195     getCellEditor : function(colIndex, rowIndex){
59196         var p = this.store.getProperty(rowIndex);
59197         var n = p.data['name'], val = p.data['value'];
59198         
59199         if(typeof(this.grid.customEditors[n]) == 'string'){
59200             return this.editors[this.grid.customEditors[n]];
59201         }
59202         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59203             return this.grid.customEditors[n];
59204         }
59205         if(val instanceof Date){
59206             return this.editors['date'];
59207         }else if(typeof val == 'number'){
59208             return this.editors['number'];
59209         }else if(typeof val == 'boolean'){
59210             return this.editors['boolean'];
59211         }else{
59212             return this.editors['string'];
59213         }
59214     }
59215 });
59216
59217 /**
59218  * @class Roo.grid.PropertyGrid
59219  * @extends Roo.grid.EditorGrid
59220  * This class represents the  interface of a component based property grid control.
59221  * <br><br>Usage:<pre><code>
59222  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59223       
59224  });
59225  // set any options
59226  grid.render();
59227  * </code></pre>
59228   
59229  * @constructor
59230  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59231  * The container MUST have some type of size defined for the grid to fill. The container will be
59232  * automatically set to position relative if it isn't already.
59233  * @param {Object} config A config object that sets properties on this grid.
59234  */
59235 Roo.grid.PropertyGrid = function(container, config){
59236     config = config || {};
59237     var store = new Roo.grid.PropertyStore(this);
59238     this.store = store;
59239     var cm = new Roo.grid.PropertyColumnModel(this, store);
59240     store.store.sort('name', 'ASC');
59241     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59242         ds: store.store,
59243         cm: cm,
59244         enableColLock:false,
59245         enableColumnMove:false,
59246         stripeRows:false,
59247         trackMouseOver: false,
59248         clicksToEdit:1
59249     }, config));
59250     this.getGridEl().addClass('x-props-grid');
59251     this.lastEditRow = null;
59252     this.on('columnresize', this.onColumnResize, this);
59253     this.addEvents({
59254          /**
59255              * @event beforepropertychange
59256              * Fires before a property changes (return false to stop?)
59257              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59258              * @param {String} id Record Id
59259              * @param {String} newval New Value
59260          * @param {String} oldval Old Value
59261              */
59262         "beforepropertychange": true,
59263         /**
59264              * @event propertychange
59265              * Fires after a property changes
59266              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59267              * @param {String} id Record Id
59268              * @param {String} newval New Value
59269          * @param {String} oldval Old Value
59270              */
59271         "propertychange": true
59272     });
59273     this.customEditors = this.customEditors || {};
59274 };
59275 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59276     
59277      /**
59278      * @cfg {Object} customEditors map of colnames=> custom editors.
59279      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59280      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59281      * false disables editing of the field.
59282          */
59283     
59284       /**
59285      * @cfg {Object} propertyNames map of property Names to their displayed value
59286          */
59287     
59288     render : function(){
59289         Roo.grid.PropertyGrid.superclass.render.call(this);
59290         this.autoSize.defer(100, this);
59291     },
59292
59293     autoSize : function(){
59294         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59295         if(this.view){
59296             this.view.fitColumns();
59297         }
59298     },
59299
59300     onColumnResize : function(){
59301         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59302         this.autoSize();
59303     },
59304     /**
59305      * Sets the data for the Grid
59306      * accepts a Key => Value object of all the elements avaiable.
59307      * @param {Object} data  to appear in grid.
59308      */
59309     setSource : function(source){
59310         this.store.setSource(source);
59311         //this.autoSize();
59312     },
59313     /**
59314      * Gets all the data from the grid.
59315      * @return {Object} data  data stored in grid
59316      */
59317     getSource : function(){
59318         return this.store.getSource();
59319     }
59320 });/*
59321   
59322  * Licence LGPL
59323  
59324  */
59325  
59326 /**
59327  * @class Roo.grid.Calendar
59328  * @extends Roo.util.Grid
59329  * This class extends the Grid to provide a calendar widget
59330  * <br><br>Usage:<pre><code>
59331  var grid = new Roo.grid.Calendar("my-container-id", {
59332      ds: myDataStore,
59333      cm: myColModel,
59334      selModel: mySelectionModel,
59335      autoSizeColumns: true,
59336      monitorWindowResize: false,
59337      trackMouseOver: true
59338      eventstore : real data store..
59339  });
59340  // set any options
59341  grid.render();
59342   
59343   * @constructor
59344  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59345  * The container MUST have some type of size defined for the grid to fill. The container will be
59346  * automatically set to position relative if it isn't already.
59347  * @param {Object} config A config object that sets properties on this grid.
59348  */
59349 Roo.grid.Calendar = function(container, config){
59350         // initialize the container
59351         this.container = Roo.get(container);
59352         this.container.update("");
59353         this.container.setStyle("overflow", "hidden");
59354     this.container.addClass('x-grid-container');
59355
59356     this.id = this.container.id;
59357
59358     Roo.apply(this, config);
59359     // check and correct shorthanded configs
59360     
59361     var rows = [];
59362     var d =1;
59363     for (var r = 0;r < 6;r++) {
59364         
59365         rows[r]=[];
59366         for (var c =0;c < 7;c++) {
59367             rows[r][c]= '';
59368         }
59369     }
59370     if (this.eventStore) {
59371         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59372         this.eventStore.on('load',this.onLoad, this);
59373         this.eventStore.on('beforeload',this.clearEvents, this);
59374          
59375     }
59376     
59377     this.dataSource = new Roo.data.Store({
59378             proxy: new Roo.data.MemoryProxy(rows),
59379             reader: new Roo.data.ArrayReader({}, [
59380                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59381     });
59382
59383     this.dataSource.load();
59384     this.ds = this.dataSource;
59385     this.ds.xmodule = this.xmodule || false;
59386     
59387     
59388     var cellRender = function(v,x,r)
59389     {
59390         return String.format(
59391             '<div class="fc-day  fc-widget-content"><div>' +
59392                 '<div class="fc-event-container"></div>' +
59393                 '<div class="fc-day-number">{0}</div>'+
59394                 
59395                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59396             '</div></div>', v);
59397     
59398     }
59399     
59400     
59401     this.colModel = new Roo.grid.ColumnModel( [
59402         {
59403             xtype: 'ColumnModel',
59404             xns: Roo.grid,
59405             dataIndex : 'weekday0',
59406             header : 'Sunday',
59407             renderer : cellRender
59408         },
59409         {
59410             xtype: 'ColumnModel',
59411             xns: Roo.grid,
59412             dataIndex : 'weekday1',
59413             header : 'Monday',
59414             renderer : cellRender
59415         },
59416         {
59417             xtype: 'ColumnModel',
59418             xns: Roo.grid,
59419             dataIndex : 'weekday2',
59420             header : 'Tuesday',
59421             renderer : cellRender
59422         },
59423         {
59424             xtype: 'ColumnModel',
59425             xns: Roo.grid,
59426             dataIndex : 'weekday3',
59427             header : 'Wednesday',
59428             renderer : cellRender
59429         },
59430         {
59431             xtype: 'ColumnModel',
59432             xns: Roo.grid,
59433             dataIndex : 'weekday4',
59434             header : 'Thursday',
59435             renderer : cellRender
59436         },
59437         {
59438             xtype: 'ColumnModel',
59439             xns: Roo.grid,
59440             dataIndex : 'weekday5',
59441             header : 'Friday',
59442             renderer : cellRender
59443         },
59444         {
59445             xtype: 'ColumnModel',
59446             xns: Roo.grid,
59447             dataIndex : 'weekday6',
59448             header : 'Saturday',
59449             renderer : cellRender
59450         }
59451     ]);
59452     this.cm = this.colModel;
59453     this.cm.xmodule = this.xmodule || false;
59454  
59455         
59456           
59457     //this.selModel = new Roo.grid.CellSelectionModel();
59458     //this.sm = this.selModel;
59459     //this.selModel.init(this);
59460     
59461     
59462     if(this.width){
59463         this.container.setWidth(this.width);
59464     }
59465
59466     if(this.height){
59467         this.container.setHeight(this.height);
59468     }
59469     /** @private */
59470         this.addEvents({
59471         // raw events
59472         /**
59473          * @event click
59474          * The raw click event for the entire grid.
59475          * @param {Roo.EventObject} e
59476          */
59477         "click" : true,
59478         /**
59479          * @event dblclick
59480          * The raw dblclick event for the entire grid.
59481          * @param {Roo.EventObject} e
59482          */
59483         "dblclick" : true,
59484         /**
59485          * @event contextmenu
59486          * The raw contextmenu event for the entire grid.
59487          * @param {Roo.EventObject} e
59488          */
59489         "contextmenu" : true,
59490         /**
59491          * @event mousedown
59492          * The raw mousedown event for the entire grid.
59493          * @param {Roo.EventObject} e
59494          */
59495         "mousedown" : true,
59496         /**
59497          * @event mouseup
59498          * The raw mouseup event for the entire grid.
59499          * @param {Roo.EventObject} e
59500          */
59501         "mouseup" : true,
59502         /**
59503          * @event mouseover
59504          * The raw mouseover event for the entire grid.
59505          * @param {Roo.EventObject} e
59506          */
59507         "mouseover" : true,
59508         /**
59509          * @event mouseout
59510          * The raw mouseout event for the entire grid.
59511          * @param {Roo.EventObject} e
59512          */
59513         "mouseout" : true,
59514         /**
59515          * @event keypress
59516          * The raw keypress event for the entire grid.
59517          * @param {Roo.EventObject} e
59518          */
59519         "keypress" : true,
59520         /**
59521          * @event keydown
59522          * The raw keydown event for the entire grid.
59523          * @param {Roo.EventObject} e
59524          */
59525         "keydown" : true,
59526
59527         // custom events
59528
59529         /**
59530          * @event cellclick
59531          * Fires when a cell is clicked
59532          * @param {Grid} this
59533          * @param {Number} rowIndex
59534          * @param {Number} columnIndex
59535          * @param {Roo.EventObject} e
59536          */
59537         "cellclick" : true,
59538         /**
59539          * @event celldblclick
59540          * Fires when a cell is double clicked
59541          * @param {Grid} this
59542          * @param {Number} rowIndex
59543          * @param {Number} columnIndex
59544          * @param {Roo.EventObject} e
59545          */
59546         "celldblclick" : true,
59547         /**
59548          * @event rowclick
59549          * Fires when a row is clicked
59550          * @param {Grid} this
59551          * @param {Number} rowIndex
59552          * @param {Roo.EventObject} e
59553          */
59554         "rowclick" : true,
59555         /**
59556          * @event rowdblclick
59557          * Fires when a row is double clicked
59558          * @param {Grid} this
59559          * @param {Number} rowIndex
59560          * @param {Roo.EventObject} e
59561          */
59562         "rowdblclick" : true,
59563         /**
59564          * @event headerclick
59565          * Fires when a header is clicked
59566          * @param {Grid} this
59567          * @param {Number} columnIndex
59568          * @param {Roo.EventObject} e
59569          */
59570         "headerclick" : true,
59571         /**
59572          * @event headerdblclick
59573          * Fires when a header cell is double clicked
59574          * @param {Grid} this
59575          * @param {Number} columnIndex
59576          * @param {Roo.EventObject} e
59577          */
59578         "headerdblclick" : true,
59579         /**
59580          * @event rowcontextmenu
59581          * Fires when a row is right clicked
59582          * @param {Grid} this
59583          * @param {Number} rowIndex
59584          * @param {Roo.EventObject} e
59585          */
59586         "rowcontextmenu" : true,
59587         /**
59588          * @event cellcontextmenu
59589          * Fires when a cell is right clicked
59590          * @param {Grid} this
59591          * @param {Number} rowIndex
59592          * @param {Number} cellIndex
59593          * @param {Roo.EventObject} e
59594          */
59595          "cellcontextmenu" : true,
59596         /**
59597          * @event headercontextmenu
59598          * Fires when a header is right clicked
59599          * @param {Grid} this
59600          * @param {Number} columnIndex
59601          * @param {Roo.EventObject} e
59602          */
59603         "headercontextmenu" : true,
59604         /**
59605          * @event bodyscroll
59606          * Fires when the body element is scrolled
59607          * @param {Number} scrollLeft
59608          * @param {Number} scrollTop
59609          */
59610         "bodyscroll" : true,
59611         /**
59612          * @event columnresize
59613          * Fires when the user resizes a column
59614          * @param {Number} columnIndex
59615          * @param {Number} newSize
59616          */
59617         "columnresize" : true,
59618         /**
59619          * @event columnmove
59620          * Fires when the user moves a column
59621          * @param {Number} oldIndex
59622          * @param {Number} newIndex
59623          */
59624         "columnmove" : true,
59625         /**
59626          * @event startdrag
59627          * Fires when row(s) start being dragged
59628          * @param {Grid} this
59629          * @param {Roo.GridDD} dd The drag drop object
59630          * @param {event} e The raw browser event
59631          */
59632         "startdrag" : true,
59633         /**
59634          * @event enddrag
59635          * Fires when a drag operation is complete
59636          * @param {Grid} this
59637          * @param {Roo.GridDD} dd The drag drop object
59638          * @param {event} e The raw browser event
59639          */
59640         "enddrag" : true,
59641         /**
59642          * @event dragdrop
59643          * Fires when dragged row(s) are dropped on a valid DD target
59644          * @param {Grid} this
59645          * @param {Roo.GridDD} dd The drag drop object
59646          * @param {String} targetId The target drag drop object
59647          * @param {event} e The raw browser event
59648          */
59649         "dragdrop" : true,
59650         /**
59651          * @event dragover
59652          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
59653          * @param {Grid} this
59654          * @param {Roo.GridDD} dd The drag drop object
59655          * @param {String} targetId The target drag drop object
59656          * @param {event} e The raw browser event
59657          */
59658         "dragover" : true,
59659         /**
59660          * @event dragenter
59661          *  Fires when the dragged row(s) first cross another DD target while being dragged
59662          * @param {Grid} this
59663          * @param {Roo.GridDD} dd The drag drop object
59664          * @param {String} targetId The target drag drop object
59665          * @param {event} e The raw browser event
59666          */
59667         "dragenter" : true,
59668         /**
59669          * @event dragout
59670          * Fires when the dragged row(s) leave another DD target while being dragged
59671          * @param {Grid} this
59672          * @param {Roo.GridDD} dd The drag drop object
59673          * @param {String} targetId The target drag drop object
59674          * @param {event} e The raw browser event
59675          */
59676         "dragout" : true,
59677         /**
59678          * @event rowclass
59679          * Fires when a row is rendered, so you can change add a style to it.
59680          * @param {GridView} gridview   The grid view
59681          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
59682          */
59683         'rowclass' : true,
59684
59685         /**
59686          * @event render
59687          * Fires when the grid is rendered
59688          * @param {Grid} grid
59689          */
59690         'render' : true,
59691             /**
59692              * @event select
59693              * Fires when a date is selected
59694              * @param {DatePicker} this
59695              * @param {Date} date The selected date
59696              */
59697         'select': true,
59698         /**
59699              * @event monthchange
59700              * Fires when the displayed month changes 
59701              * @param {DatePicker} this
59702              * @param {Date} date The selected month
59703              */
59704         'monthchange': true,
59705         /**
59706              * @event evententer
59707              * Fires when mouse over an event
59708              * @param {Calendar} this
59709              * @param {event} Event
59710              */
59711         'evententer': true,
59712         /**
59713              * @event eventleave
59714              * Fires when the mouse leaves an
59715              * @param {Calendar} this
59716              * @param {event}
59717              */
59718         'eventleave': true,
59719         /**
59720              * @event eventclick
59721              * Fires when the mouse click an
59722              * @param {Calendar} this
59723              * @param {event}
59724              */
59725         'eventclick': true,
59726         /**
59727              * @event eventrender
59728              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
59729              * @param {Calendar} this
59730              * @param {data} data to be modified
59731              */
59732         'eventrender': true
59733         
59734     });
59735
59736     Roo.grid.Grid.superclass.constructor.call(this);
59737     this.on('render', function() {
59738         this.view.el.addClass('x-grid-cal'); 
59739         
59740         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
59741
59742     },this);
59743     
59744     if (!Roo.grid.Calendar.style) {
59745         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
59746             
59747             
59748             '.x-grid-cal .x-grid-col' :  {
59749                 height: 'auto !important',
59750                 'vertical-align': 'top'
59751             },
59752             '.x-grid-cal  .fc-event-hori' : {
59753                 height: '14px'
59754             }
59755              
59756             
59757         }, Roo.id());
59758     }
59759
59760     
59761     
59762 };
59763 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
59764     /**
59765      * @cfg {Store} eventStore The store that loads events.
59766      */
59767     eventStore : 25,
59768
59769      
59770     activeDate : false,
59771     startDay : 0,
59772     autoWidth : true,
59773     monitorWindowResize : false,
59774
59775     
59776     resizeColumns : function() {
59777         var col = (this.view.el.getWidth() / 7) - 3;
59778         // loop through cols, and setWidth
59779         for(var i =0 ; i < 7 ; i++){
59780             this.cm.setColumnWidth(i, col);
59781         }
59782     },
59783      setDate :function(date) {
59784         
59785         Roo.log('setDate?');
59786         
59787         this.resizeColumns();
59788         var vd = this.activeDate;
59789         this.activeDate = date;
59790 //        if(vd && this.el){
59791 //            var t = date.getTime();
59792 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
59793 //                Roo.log('using add remove');
59794 //                
59795 //                this.fireEvent('monthchange', this, date);
59796 //                
59797 //                this.cells.removeClass("fc-state-highlight");
59798 //                this.cells.each(function(c){
59799 //                   if(c.dateValue == t){
59800 //                       c.addClass("fc-state-highlight");
59801 //                       setTimeout(function(){
59802 //                            try{c.dom.firstChild.focus();}catch(e){}
59803 //                       }, 50);
59804 //                       return false;
59805 //                   }
59806 //                   return true;
59807 //                });
59808 //                return;
59809 //            }
59810 //        }
59811         
59812         var days = date.getDaysInMonth();
59813         
59814         var firstOfMonth = date.getFirstDateOfMonth();
59815         var startingPos = firstOfMonth.getDay()-this.startDay;
59816         
59817         if(startingPos < this.startDay){
59818             startingPos += 7;
59819         }
59820         
59821         var pm = date.add(Date.MONTH, -1);
59822         var prevStart = pm.getDaysInMonth()-startingPos;
59823 //        
59824         
59825         
59826         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59827         
59828         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
59829         //this.cells.addClassOnOver('fc-state-hover');
59830         
59831         var cells = this.cells.elements;
59832         var textEls = this.textNodes;
59833         
59834         //Roo.each(cells, function(cell){
59835         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
59836         //});
59837         
59838         days += startingPos;
59839
59840         // convert everything to numbers so it's fast
59841         var day = 86400000;
59842         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
59843         //Roo.log(d);
59844         //Roo.log(pm);
59845         //Roo.log(prevStart);
59846         
59847         var today = new Date().clearTime().getTime();
59848         var sel = date.clearTime().getTime();
59849         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
59850         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
59851         var ddMatch = this.disabledDatesRE;
59852         var ddText = this.disabledDatesText;
59853         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
59854         var ddaysText = this.disabledDaysText;
59855         var format = this.format;
59856         
59857         var setCellClass = function(cal, cell){
59858             
59859             //Roo.log('set Cell Class');
59860             cell.title = "";
59861             var t = d.getTime();
59862             
59863             //Roo.log(d);
59864             
59865             
59866             cell.dateValue = t;
59867             if(t == today){
59868                 cell.className += " fc-today";
59869                 cell.className += " fc-state-highlight";
59870                 cell.title = cal.todayText;
59871             }
59872             if(t == sel){
59873                 // disable highlight in other month..
59874                 cell.className += " fc-state-highlight";
59875                 
59876             }
59877             // disabling
59878             if(t < min) {
59879                 //cell.className = " fc-state-disabled";
59880                 cell.title = cal.minText;
59881                 return;
59882             }
59883             if(t > max) {
59884                 //cell.className = " fc-state-disabled";
59885                 cell.title = cal.maxText;
59886                 return;
59887             }
59888             if(ddays){
59889                 if(ddays.indexOf(d.getDay()) != -1){
59890                     // cell.title = ddaysText;
59891                    // cell.className = " fc-state-disabled";
59892                 }
59893             }
59894             if(ddMatch && format){
59895                 var fvalue = d.dateFormat(format);
59896                 if(ddMatch.test(fvalue)){
59897                     cell.title = ddText.replace("%0", fvalue);
59898                    cell.className = " fc-state-disabled";
59899                 }
59900             }
59901             
59902             if (!cell.initialClassName) {
59903                 cell.initialClassName = cell.dom.className;
59904             }
59905             
59906             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
59907         };
59908
59909         var i = 0;
59910         
59911         for(; i < startingPos; i++) {
59912             cells[i].dayName =  (++prevStart);
59913             Roo.log(textEls[i]);
59914             d.setDate(d.getDate()+1);
59915             
59916             //cells[i].className = "fc-past fc-other-month";
59917             setCellClass(this, cells[i]);
59918         }
59919         
59920         var intDay = 0;
59921         
59922         for(; i < days; i++){
59923             intDay = i - startingPos + 1;
59924             cells[i].dayName =  (intDay);
59925             d.setDate(d.getDate()+1);
59926             
59927             cells[i].className = ''; // "x-date-active";
59928             setCellClass(this, cells[i]);
59929         }
59930         var extraDays = 0;
59931         
59932         for(; i < 42; i++) {
59933             //textEls[i].innerHTML = (++extraDays);
59934             
59935             d.setDate(d.getDate()+1);
59936             cells[i].dayName = (++extraDays);
59937             cells[i].className = "fc-future fc-other-month";
59938             setCellClass(this, cells[i]);
59939         }
59940         
59941         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
59942         
59943         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
59944         
59945         // this will cause all the cells to mis
59946         var rows= [];
59947         var i =0;
59948         for (var r = 0;r < 6;r++) {
59949             for (var c =0;c < 7;c++) {
59950                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
59951             }    
59952         }
59953         
59954         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59955         for(i=0;i<cells.length;i++) {
59956             
59957             this.cells.elements[i].dayName = cells[i].dayName ;
59958             this.cells.elements[i].className = cells[i].className;
59959             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
59960             this.cells.elements[i].title = cells[i].title ;
59961             this.cells.elements[i].dateValue = cells[i].dateValue ;
59962         }
59963         
59964         
59965         
59966         
59967         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
59968         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
59969         
59970         ////if(totalRows != 6){
59971             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
59972            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
59973        // }
59974         
59975         this.fireEvent('monthchange', this, date);
59976         
59977         
59978     },
59979  /**
59980      * Returns the grid's SelectionModel.
59981      * @return {SelectionModel}
59982      */
59983     getSelectionModel : function(){
59984         if(!this.selModel){
59985             this.selModel = new Roo.grid.CellSelectionModel();
59986         }
59987         return this.selModel;
59988     },
59989
59990     load: function() {
59991         this.eventStore.load()
59992         
59993         
59994         
59995     },
59996     
59997     findCell : function(dt) {
59998         dt = dt.clearTime().getTime();
59999         var ret = false;
60000         this.cells.each(function(c){
60001             //Roo.log("check " +c.dateValue + '?=' + dt);
60002             if(c.dateValue == dt){
60003                 ret = c;
60004                 return false;
60005             }
60006             return true;
60007         });
60008         
60009         return ret;
60010     },
60011     
60012     findCells : function(rec) {
60013         var s = rec.data.start_dt.clone().clearTime().getTime();
60014        // Roo.log(s);
60015         var e= rec.data.end_dt.clone().clearTime().getTime();
60016        // Roo.log(e);
60017         var ret = [];
60018         this.cells.each(function(c){
60019              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60020             
60021             if(c.dateValue > e){
60022                 return ;
60023             }
60024             if(c.dateValue < s){
60025                 return ;
60026             }
60027             ret.push(c);
60028         });
60029         
60030         return ret;    
60031     },
60032     
60033     findBestRow: function(cells)
60034     {
60035         var ret = 0;
60036         
60037         for (var i =0 ; i < cells.length;i++) {
60038             ret  = Math.max(cells[i].rows || 0,ret);
60039         }
60040         return ret;
60041         
60042     },
60043     
60044     
60045     addItem : function(rec)
60046     {
60047         // look for vertical location slot in
60048         var cells = this.findCells(rec);
60049         
60050         rec.row = this.findBestRow(cells);
60051         
60052         // work out the location.
60053         
60054         var crow = false;
60055         var rows = [];
60056         for(var i =0; i < cells.length; i++) {
60057             if (!crow) {
60058                 crow = {
60059                     start : cells[i],
60060                     end :  cells[i]
60061                 };
60062                 continue;
60063             }
60064             if (crow.start.getY() == cells[i].getY()) {
60065                 // on same row.
60066                 crow.end = cells[i];
60067                 continue;
60068             }
60069             // different row.
60070             rows.push(crow);
60071             crow = {
60072                 start: cells[i],
60073                 end : cells[i]
60074             };
60075             
60076         }
60077         
60078         rows.push(crow);
60079         rec.els = [];
60080         rec.rows = rows;
60081         rec.cells = cells;
60082         for (var i = 0; i < cells.length;i++) {
60083             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60084             
60085         }
60086         
60087         
60088     },
60089     
60090     clearEvents: function() {
60091         
60092         if (!this.eventStore.getCount()) {
60093             return;
60094         }
60095         // reset number of rows in cells.
60096         Roo.each(this.cells.elements, function(c){
60097             c.rows = 0;
60098         });
60099         
60100         this.eventStore.each(function(e) {
60101             this.clearEvent(e);
60102         },this);
60103         
60104     },
60105     
60106     clearEvent : function(ev)
60107     {
60108         if (ev.els) {
60109             Roo.each(ev.els, function(el) {
60110                 el.un('mouseenter' ,this.onEventEnter, this);
60111                 el.un('mouseleave' ,this.onEventLeave, this);
60112                 el.remove();
60113             },this);
60114             ev.els = [];
60115         }
60116     },
60117     
60118     
60119     renderEvent : function(ev,ctr) {
60120         if (!ctr) {
60121              ctr = this.view.el.select('.fc-event-container',true).first();
60122         }
60123         
60124          
60125         this.clearEvent(ev);
60126             //code
60127        
60128         
60129         
60130         ev.els = [];
60131         var cells = ev.cells;
60132         var rows = ev.rows;
60133         this.fireEvent('eventrender', this, ev);
60134         
60135         for(var i =0; i < rows.length; i++) {
60136             
60137             cls = '';
60138             if (i == 0) {
60139                 cls += ' fc-event-start';
60140             }
60141             if ((i+1) == rows.length) {
60142                 cls += ' fc-event-end';
60143             }
60144             
60145             //Roo.log(ev.data);
60146             // how many rows should it span..
60147             var cg = this.eventTmpl.append(ctr,Roo.apply({
60148                 fccls : cls
60149                 
60150             }, ev.data) , true);
60151             
60152             
60153             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60154             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60155             cg.on('click', this.onEventClick, this, ev);
60156             
60157             ev.els.push(cg);
60158             
60159             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60160             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60161             //Roo.log(cg);
60162              
60163             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60164             cg.setWidth(ebox.right - sbox.x -2);
60165         }
60166     },
60167     
60168     renderEvents: function()
60169     {   
60170         // first make sure there is enough space..
60171         
60172         if (!this.eventTmpl) {
60173             this.eventTmpl = new Roo.Template(
60174                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60175                     '<div class="fc-event-inner">' +
60176                         '<span class="fc-event-time">{time}</span>' +
60177                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60178                     '</div>' +
60179                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60180                 '</div>'
60181             );
60182                 
60183         }
60184                
60185         
60186         
60187         this.cells.each(function(c) {
60188             //Roo.log(c.select('.fc-day-content div',true).first());
60189             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60190         });
60191         
60192         var ctr = this.view.el.select('.fc-event-container',true).first();
60193         
60194         var cls;
60195         this.eventStore.each(function(ev){
60196             
60197             this.renderEvent(ev);
60198              
60199              
60200         }, this);
60201         this.view.layout();
60202         
60203     },
60204     
60205     onEventEnter: function (e, el,event,d) {
60206         this.fireEvent('evententer', this, el, event);
60207     },
60208     
60209     onEventLeave: function (e, el,event,d) {
60210         this.fireEvent('eventleave', this, el, event);
60211     },
60212     
60213     onEventClick: function (e, el,event,d) {
60214         this.fireEvent('eventclick', this, el, event);
60215     },
60216     
60217     onMonthChange: function () {
60218         this.store.load();
60219     },
60220     
60221     onLoad: function () {
60222         
60223         //Roo.log('calendar onload');
60224 //         
60225         if(this.eventStore.getCount() > 0){
60226             
60227            
60228             
60229             this.eventStore.each(function(d){
60230                 
60231                 
60232                 // FIXME..
60233                 var add =   d.data;
60234                 if (typeof(add.end_dt) == 'undefined')  {
60235                     Roo.log("Missing End time in calendar data: ");
60236                     Roo.log(d);
60237                     return;
60238                 }
60239                 if (typeof(add.start_dt) == 'undefined')  {
60240                     Roo.log("Missing Start time in calendar data: ");
60241                     Roo.log(d);
60242                     return;
60243                 }
60244                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60245                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60246                 add.id = add.id || d.id;
60247                 add.title = add.title || '??';
60248                 
60249                 this.addItem(d);
60250                 
60251              
60252             },this);
60253         }
60254         
60255         this.renderEvents();
60256     }
60257     
60258
60259 });
60260 /*
60261  grid : {
60262                 xtype: 'Grid',
60263                 xns: Roo.grid,
60264                 listeners : {
60265                     render : function ()
60266                     {
60267                         _this.grid = this;
60268                         
60269                         if (!this.view.el.hasClass('course-timesheet')) {
60270                             this.view.el.addClass('course-timesheet');
60271                         }
60272                         if (this.tsStyle) {
60273                             this.ds.load({});
60274                             return; 
60275                         }
60276                         Roo.log('width');
60277                         Roo.log(_this.grid.view.el.getWidth());
60278                         
60279                         
60280                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60281                             '.course-timesheet .x-grid-row' : {
60282                                 height: '80px'
60283                             },
60284                             '.x-grid-row td' : {
60285                                 'vertical-align' : 0
60286                             },
60287                             '.course-edit-link' : {
60288                                 'color' : 'blue',
60289                                 'text-overflow' : 'ellipsis',
60290                                 'overflow' : 'hidden',
60291                                 'white-space' : 'nowrap',
60292                                 'cursor' : 'pointer'
60293                             },
60294                             '.sub-link' : {
60295                                 'color' : 'green'
60296                             },
60297                             '.de-act-sup-link' : {
60298                                 'color' : 'purple',
60299                                 'text-decoration' : 'line-through'
60300                             },
60301                             '.de-act-link' : {
60302                                 'color' : 'red',
60303                                 'text-decoration' : 'line-through'
60304                             },
60305                             '.course-timesheet .course-highlight' : {
60306                                 'border-top-style': 'dashed !important',
60307                                 'border-bottom-bottom': 'dashed !important'
60308                             },
60309                             '.course-timesheet .course-item' : {
60310                                 'font-family'   : 'tahoma, arial, helvetica',
60311                                 'font-size'     : '11px',
60312                                 'overflow'      : 'hidden',
60313                                 'padding-left'  : '10px',
60314                                 'padding-right' : '10px',
60315                                 'padding-top' : '10px' 
60316                             }
60317                             
60318                         }, Roo.id());
60319                                 this.ds.load({});
60320                     }
60321                 },
60322                 autoWidth : true,
60323                 monitorWindowResize : false,
60324                 cellrenderer : function(v,x,r)
60325                 {
60326                     return v;
60327                 },
60328                 sm : {
60329                     xtype: 'CellSelectionModel',
60330                     xns: Roo.grid
60331                 },
60332                 dataSource : {
60333                     xtype: 'Store',
60334                     xns: Roo.data,
60335                     listeners : {
60336                         beforeload : function (_self, options)
60337                         {
60338                             options.params = options.params || {};
60339                             options.params._month = _this.monthField.getValue();
60340                             options.params.limit = 9999;
60341                             options.params['sort'] = 'when_dt';    
60342                             options.params['dir'] = 'ASC';    
60343                             this.proxy.loadResponse = this.loadResponse;
60344                             Roo.log("load?");
60345                             //this.addColumns();
60346                         },
60347                         load : function (_self, records, options)
60348                         {
60349                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60350                                 // if you click on the translation.. you can edit it...
60351                                 var el = Roo.get(this);
60352                                 var id = el.dom.getAttribute('data-id');
60353                                 var d = el.dom.getAttribute('data-date');
60354                                 var t = el.dom.getAttribute('data-time');
60355                                 //var id = this.child('span').dom.textContent;
60356                                 
60357                                 //Roo.log(this);
60358                                 Pman.Dialog.CourseCalendar.show({
60359                                     id : id,
60360                                     when_d : d,
60361                                     when_t : t,
60362                                     productitem_active : id ? 1 : 0
60363                                 }, function() {
60364                                     _this.grid.ds.load({});
60365                                 });
60366                            
60367                            });
60368                            
60369                            _this.panel.fireEvent('resize', [ '', '' ]);
60370                         }
60371                     },
60372                     loadResponse : function(o, success, response){
60373                             // this is overridden on before load..
60374                             
60375                             Roo.log("our code?");       
60376                             //Roo.log(success);
60377                             //Roo.log(response)
60378                             delete this.activeRequest;
60379                             if(!success){
60380                                 this.fireEvent("loadexception", this, o, response);
60381                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60382                                 return;
60383                             }
60384                             var result;
60385                             try {
60386                                 result = o.reader.read(response);
60387                             }catch(e){
60388                                 Roo.log("load exception?");
60389                                 this.fireEvent("loadexception", this, o, response, e);
60390                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60391                                 return;
60392                             }
60393                             Roo.log("ready...");        
60394                             // loop through result.records;
60395                             // and set this.tdate[date] = [] << array of records..
60396                             _this.tdata  = {};
60397                             Roo.each(result.records, function(r){
60398                                 //Roo.log(r.data);
60399                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60400                                     _this.tdata[r.data.when_dt.format('j')] = [];
60401                                 }
60402                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60403                             });
60404                             
60405                             //Roo.log(_this.tdata);
60406                             
60407                             result.records = [];
60408                             result.totalRecords = 6;
60409                     
60410                             // let's generate some duumy records for the rows.
60411                             //var st = _this.dateField.getValue();
60412                             
60413                             // work out monday..
60414                             //st = st.add(Date.DAY, -1 * st.format('w'));
60415                             
60416                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60417                             
60418                             var firstOfMonth = date.getFirstDayOfMonth();
60419                             var days = date.getDaysInMonth();
60420                             var d = 1;
60421                             var firstAdded = false;
60422                             for (var i = 0; i < result.totalRecords ; i++) {
60423                                 //var d= st.add(Date.DAY, i);
60424                                 var row = {};
60425                                 var added = 0;
60426                                 for(var w = 0 ; w < 7 ; w++){
60427                                     if(!firstAdded && firstOfMonth != w){
60428                                         continue;
60429                                     }
60430                                     if(d > days){
60431                                         continue;
60432                                     }
60433                                     firstAdded = true;
60434                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60435                                     row['weekday'+w] = String.format(
60436                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60437                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60438                                                     d,
60439                                                     date.format('Y-m-')+dd
60440                                                 );
60441                                     added++;
60442                                     if(typeof(_this.tdata[d]) != 'undefined'){
60443                                         Roo.each(_this.tdata[d], function(r){
60444                                             var is_sub = '';
60445                                             var deactive = '';
60446                                             var id = r.id;
60447                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60448                                             if(r.parent_id*1>0){
60449                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60450                                                 id = r.parent_id;
60451                                             }
60452                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60453                                                 deactive = 'de-act-link';
60454                                             }
60455                                             
60456                                             row['weekday'+w] += String.format(
60457                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60458                                                     id, //0
60459                                                     r.product_id_name, //1
60460                                                     r.when_dt.format('h:ia'), //2
60461                                                     is_sub, //3
60462                                                     deactive, //4
60463                                                     desc // 5
60464                                             );
60465                                         });
60466                                     }
60467                                     d++;
60468                                 }
60469                                 
60470                                 // only do this if something added..
60471                                 if(added > 0){ 
60472                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60473                                 }
60474                                 
60475                                 
60476                                 // push it twice. (second one with an hour..
60477                                 
60478                             }
60479                             //Roo.log(result);
60480                             this.fireEvent("load", this, o, o.request.arg);
60481                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60482                         },
60483                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60484                     proxy : {
60485                         xtype: 'HttpProxy',
60486                         xns: Roo.data,
60487                         method : 'GET',
60488                         url : baseURL + '/Roo/Shop_course.php'
60489                     },
60490                     reader : {
60491                         xtype: 'JsonReader',
60492                         xns: Roo.data,
60493                         id : 'id',
60494                         fields : [
60495                             {
60496                                 'name': 'id',
60497                                 'type': 'int'
60498                             },
60499                             {
60500                                 'name': 'when_dt',
60501                                 'type': 'string'
60502                             },
60503                             {
60504                                 'name': 'end_dt',
60505                                 'type': 'string'
60506                             },
60507                             {
60508                                 'name': 'parent_id',
60509                                 'type': 'int'
60510                             },
60511                             {
60512                                 'name': 'product_id',
60513                                 'type': 'int'
60514                             },
60515                             {
60516                                 'name': 'productitem_id',
60517                                 'type': 'int'
60518                             },
60519                             {
60520                                 'name': 'guid',
60521                                 'type': 'int'
60522                             }
60523                         ]
60524                     }
60525                 },
60526                 toolbar : {
60527                     xtype: 'Toolbar',
60528                     xns: Roo,
60529                     items : [
60530                         {
60531                             xtype: 'Button',
60532                             xns: Roo.Toolbar,
60533                             listeners : {
60534                                 click : function (_self, e)
60535                                 {
60536                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60537                                     sd.setMonth(sd.getMonth()-1);
60538                                     _this.monthField.setValue(sd.format('Y-m-d'));
60539                                     _this.grid.ds.load({});
60540                                 }
60541                             },
60542                             text : "Back"
60543                         },
60544                         {
60545                             xtype: 'Separator',
60546                             xns: Roo.Toolbar
60547                         },
60548                         {
60549                             xtype: 'MonthField',
60550                             xns: Roo.form,
60551                             listeners : {
60552                                 render : function (_self)
60553                                 {
60554                                     _this.monthField = _self;
60555                                    // _this.monthField.set  today
60556                                 },
60557                                 select : function (combo, date)
60558                                 {
60559                                     _this.grid.ds.load({});
60560                                 }
60561                             },
60562                             value : (function() { return new Date(); })()
60563                         },
60564                         {
60565                             xtype: 'Separator',
60566                             xns: Roo.Toolbar
60567                         },
60568                         {
60569                             xtype: 'TextItem',
60570                             xns: Roo.Toolbar,
60571                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60572                         },
60573                         {
60574                             xtype: 'Fill',
60575                             xns: Roo.Toolbar
60576                         },
60577                         {
60578                             xtype: 'Button',
60579                             xns: Roo.Toolbar,
60580                             listeners : {
60581                                 click : function (_self, e)
60582                                 {
60583                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60584                                     sd.setMonth(sd.getMonth()+1);
60585                                     _this.monthField.setValue(sd.format('Y-m-d'));
60586                                     _this.grid.ds.load({});
60587                                 }
60588                             },
60589                             text : "Next"
60590                         }
60591                     ]
60592                 },
60593                  
60594             }
60595         };
60596         
60597         *//*
60598  * Based on:
60599  * Ext JS Library 1.1.1
60600  * Copyright(c) 2006-2007, Ext JS, LLC.
60601  *
60602  * Originally Released Under LGPL - original licence link has changed is not relivant.
60603  *
60604  * Fork - LGPL
60605  * <script type="text/javascript">
60606  */
60607  
60608 /**
60609  * @class Roo.LoadMask
60610  * A simple utility class for generically masking elements while loading data.  If the element being masked has
60611  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
60612  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
60613  * element's UpdateManager load indicator and will be destroyed after the initial load.
60614  * @constructor
60615  * Create a new LoadMask
60616  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
60617  * @param {Object} config The config object
60618  */
60619 Roo.LoadMask = function(el, config){
60620     this.el = Roo.get(el);
60621     Roo.apply(this, config);
60622     if(this.store){
60623         this.store.on('beforeload', this.onBeforeLoad, this);
60624         this.store.on('load', this.onLoad, this);
60625         this.store.on('loadexception', this.onLoadException, this);
60626         this.removeMask = false;
60627     }else{
60628         var um = this.el.getUpdateManager();
60629         um.showLoadIndicator = false; // disable the default indicator
60630         um.on('beforeupdate', this.onBeforeLoad, this);
60631         um.on('update', this.onLoad, this);
60632         um.on('failure', this.onLoad, this);
60633         this.removeMask = true;
60634     }
60635 };
60636
60637 Roo.LoadMask.prototype = {
60638     /**
60639      * @cfg {Boolean} removeMask
60640      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
60641      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
60642      */
60643     /**
60644      * @cfg {String} msg
60645      * The text to display in a centered loading message box (defaults to 'Loading...')
60646      */
60647     msg : 'Loading...',
60648     /**
60649      * @cfg {String} msgCls
60650      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
60651      */
60652     msgCls : 'x-mask-loading',
60653
60654     /**
60655      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
60656      * @type Boolean
60657      */
60658     disabled: false,
60659
60660     /**
60661      * Disables the mask to prevent it from being displayed
60662      */
60663     disable : function(){
60664        this.disabled = true;
60665     },
60666
60667     /**
60668      * Enables the mask so that it can be displayed
60669      */
60670     enable : function(){
60671         this.disabled = false;
60672     },
60673     
60674     onLoadException : function()
60675     {
60676         Roo.log(arguments);
60677         
60678         if (typeof(arguments[3]) != 'undefined') {
60679             Roo.MessageBox.alert("Error loading",arguments[3]);
60680         } 
60681         /*
60682         try {
60683             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
60684                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
60685             }   
60686         } catch(e) {
60687             
60688         }
60689         */
60690     
60691         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60692     },
60693     // private
60694     onLoad : function()
60695     {
60696         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60697     },
60698
60699     // private
60700     onBeforeLoad : function(){
60701         if(!this.disabled){
60702             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
60703         }
60704     },
60705
60706     // private
60707     destroy : function(){
60708         if(this.store){
60709             this.store.un('beforeload', this.onBeforeLoad, this);
60710             this.store.un('load', this.onLoad, this);
60711             this.store.un('loadexception', this.onLoadException, this);
60712         }else{
60713             var um = this.el.getUpdateManager();
60714             um.un('beforeupdate', this.onBeforeLoad, this);
60715             um.un('update', this.onLoad, this);
60716             um.un('failure', this.onLoad, this);
60717         }
60718     }
60719 };/*
60720  * Based on:
60721  * Ext JS Library 1.1.1
60722  * Copyright(c) 2006-2007, Ext JS, LLC.
60723  *
60724  * Originally Released Under LGPL - original licence link has changed is not relivant.
60725  *
60726  * Fork - LGPL
60727  * <script type="text/javascript">
60728  */
60729
60730
60731 /**
60732  * @class Roo.XTemplate
60733  * @extends Roo.Template
60734  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
60735 <pre><code>
60736 var t = new Roo.XTemplate(
60737         '&lt;select name="{name}"&gt;',
60738                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
60739         '&lt;/select&gt;'
60740 );
60741  
60742 // then append, applying the master template values
60743  </code></pre>
60744  *
60745  * Supported features:
60746  *
60747  *  Tags:
60748
60749 <pre><code>
60750       {a_variable} - output encoded.
60751       {a_variable.format:("Y-m-d")} - call a method on the variable
60752       {a_variable:raw} - unencoded output
60753       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
60754       {a_variable:this.method_on_template(...)} - call a method on the template object.
60755  
60756 </code></pre>
60757  *  The tpl tag:
60758 <pre><code>
60759         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
60760         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
60761         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
60762         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
60763   
60764         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
60765         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
60766 </code></pre>
60767  *      
60768  */
60769 Roo.XTemplate = function()
60770 {
60771     Roo.XTemplate.superclass.constructor.apply(this, arguments);
60772     if (this.html) {
60773         this.compile();
60774     }
60775 };
60776
60777
60778 Roo.extend(Roo.XTemplate, Roo.Template, {
60779
60780     /**
60781      * The various sub templates
60782      */
60783     tpls : false,
60784     /**
60785      *
60786      * basic tag replacing syntax
60787      * WORD:WORD()
60788      *
60789      * // you can fake an object call by doing this
60790      *  x.t:(test,tesT) 
60791      * 
60792      */
60793     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
60794
60795     /**
60796      * compile the template
60797      *
60798      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
60799      *
60800      */
60801     compile: function()
60802     {
60803         var s = this.html;
60804      
60805         s = ['<tpl>', s, '</tpl>'].join('');
60806     
60807         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
60808             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
60809             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
60810             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
60811             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
60812             m,
60813             id     = 0,
60814             tpls   = [];
60815     
60816         while(true == !!(m = s.match(re))){
60817             var forMatch   = m[0].match(nameRe),
60818                 ifMatch   = m[0].match(ifRe),
60819                 execMatch   = m[0].match(execRe),
60820                 namedMatch   = m[0].match(namedRe),
60821                 
60822                 exp  = null, 
60823                 fn   = null,
60824                 exec = null,
60825                 name = forMatch && forMatch[1] ? forMatch[1] : '';
60826                 
60827             if (ifMatch) {
60828                 // if - puts fn into test..
60829                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
60830                 if(exp){
60831                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
60832                 }
60833             }
60834             
60835             if (execMatch) {
60836                 // exec - calls a function... returns empty if true is  returned.
60837                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
60838                 if(exp){
60839                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
60840                 }
60841             }
60842             
60843             
60844             if (name) {
60845                 // for = 
60846                 switch(name){
60847                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
60848                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
60849                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
60850                 }
60851             }
60852             var uid = namedMatch ? namedMatch[1] : id;
60853             
60854             
60855             tpls.push({
60856                 id:     namedMatch ? namedMatch[1] : id,
60857                 target: name,
60858                 exec:   exec,
60859                 test:   fn,
60860                 body:   m[1] || ''
60861             });
60862             if (namedMatch) {
60863                 s = s.replace(m[0], '');
60864             } else { 
60865                 s = s.replace(m[0], '{xtpl'+ id + '}');
60866             }
60867             ++id;
60868         }
60869         this.tpls = [];
60870         for(var i = tpls.length-1; i >= 0; --i){
60871             this.compileTpl(tpls[i]);
60872             this.tpls[tpls[i].id] = tpls[i];
60873         }
60874         this.master = tpls[tpls.length-1];
60875         return this;
60876     },
60877     /**
60878      * same as applyTemplate, except it's done to one of the subTemplates
60879      * when using named templates, you can do:
60880      *
60881      * var str = pl.applySubTemplate('your-name', values);
60882      *
60883      * 
60884      * @param {Number} id of the template
60885      * @param {Object} values to apply to template
60886      * @param {Object} parent (normaly the instance of this object)
60887      */
60888     applySubTemplate : function(id, values, parent)
60889     {
60890         
60891         
60892         var t = this.tpls[id];
60893         
60894         
60895         try { 
60896             if(t.test && !t.test.call(this, values, parent)){
60897                 return '';
60898             }
60899         } catch(e) {
60900             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
60901             Roo.log(e.toString());
60902             Roo.log(t.test);
60903             return ''
60904         }
60905         try { 
60906             
60907             if(t.exec && t.exec.call(this, values, parent)){
60908                 return '';
60909             }
60910         } catch(e) {
60911             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
60912             Roo.log(e.toString());
60913             Roo.log(t.exec);
60914             return ''
60915         }
60916         try {
60917             var vs = t.target ? t.target.call(this, values, parent) : values;
60918             parent = t.target ? values : parent;
60919             if(t.target && vs instanceof Array){
60920                 var buf = [];
60921                 for(var i = 0, len = vs.length; i < len; i++){
60922                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
60923                 }
60924                 return buf.join('');
60925             }
60926             return t.compiled.call(this, vs, parent);
60927         } catch (e) {
60928             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
60929             Roo.log(e.toString());
60930             Roo.log(t.compiled);
60931             return '';
60932         }
60933     },
60934
60935     compileTpl : function(tpl)
60936     {
60937         var fm = Roo.util.Format;
60938         var useF = this.disableFormats !== true;
60939         var sep = Roo.isGecko ? "+" : ",";
60940         var undef = function(str) {
60941             Roo.log("Property not found :"  + str);
60942             return '';
60943         };
60944         
60945         var fn = function(m, name, format, args)
60946         {
60947             //Roo.log(arguments);
60948             args = args ? args.replace(/\\'/g,"'") : args;
60949             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
60950             if (typeof(format) == 'undefined') {
60951                 format= 'htmlEncode';
60952             }
60953             if (format == 'raw' ) {
60954                 format = false;
60955             }
60956             
60957             if(name.substr(0, 4) == 'xtpl'){
60958                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
60959             }
60960             
60961             // build an array of options to determine if value is undefined..
60962             
60963             // basically get 'xxxx.yyyy' then do
60964             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
60965             //    (function () { Roo.log("Property not found"); return ''; })() :
60966             //    ......
60967             
60968             var udef_ar = [];
60969             var lookfor = '';
60970             Roo.each(name.split('.'), function(st) {
60971                 lookfor += (lookfor.length ? '.': '') + st;
60972                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
60973             });
60974             
60975             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
60976             
60977             
60978             if(format && useF){
60979                 
60980                 args = args ? ',' + args : "";
60981                  
60982                 if(format.substr(0, 5) != "this."){
60983                     format = "fm." + format + '(';
60984                 }else{
60985                     format = 'this.call("'+ format.substr(5) + '", ';
60986                     args = ", values";
60987                 }
60988                 
60989                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
60990             }
60991              
60992             if (args.length) {
60993                 // called with xxyx.yuu:(test,test)
60994                 // change to ()
60995                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
60996             }
60997             // raw.. - :raw modifier..
60998             return "'"+ sep + udef_st  + name + ")"+sep+"'";
60999             
61000         };
61001         var body;
61002         // branched to use + in gecko and [].join() in others
61003         if(Roo.isGecko){
61004             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61005                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61006                     "';};};";
61007         }else{
61008             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61009             body.push(tpl.body.replace(/(\r\n|\n)/g,
61010                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61011             body.push("'].join('');};};");
61012             body = body.join('');
61013         }
61014         
61015         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61016        
61017         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61018         eval(body);
61019         
61020         return this;
61021     },
61022
61023     applyTemplate : function(values){
61024         return this.master.compiled.call(this, values, {});
61025         //var s = this.subs;
61026     },
61027
61028     apply : function(){
61029         return this.applyTemplate.apply(this, arguments);
61030     }
61031
61032  });
61033
61034 Roo.XTemplate.from = function(el){
61035     el = Roo.getDom(el);
61036     return new Roo.XTemplate(el.value || el.innerHTML);
61037 };