Roo/form/ComboNested.js
[roojs1] / roojs-debug.js
1 /*
2  * Based on:
3  * Ext JS Library 1.1.1
4  * Copyright(c) 2006-2007, Ext JS, LLC.
5  *
6  * Originally Released Under LGPL - original licence link has changed is not relivant.
7  *
8  * Fork - LGPL
9  * <script type="text/javascript">
10  */
11  
12
13
14
15
16 // for old browsers
17 window["undefined"] = window["undefined"];
18
19 /**
20  * @class Roo
21  * Roo core utilities and functions.
22  * @singleton
23  */
24 var Roo = {}; 
25 /**
26  * Copies all the properties of config to obj.
27  * @param {Object} obj The receiver of the properties
28  * @param {Object} config The source of the properties
29  * @param {Object} defaults A different object that will also be applied for default values
30  * @return {Object} returns obj
31  * @member Roo apply
32  */
33
34  
35 Roo.apply = function(o, c, defaults){
36     if(defaults){
37         // no "this" reference for friendly out of scope calls
38         Roo.apply(o, defaults);
39     }
40     if(o && c && typeof c == 'object'){
41         for(var p in c){
42             o[p] = c[p];
43         }
44     }
45     return o;
46 };
47
48
49 (function(){
50     var idSeed = 0;
51     var ua = navigator.userAgent.toLowerCase();
52
53     var isStrict = document.compatMode == "CSS1Compat",
54         isOpera = ua.indexOf("opera") > -1,
55         isSafari = (/webkit|khtml/).test(ua),
56         isFirefox = ua.indexOf("firefox") > -1,
57         isIE = ua.indexOf("msie") > -1,
58         isIE7 = ua.indexOf("msie 7") > -1,
59         isIE11 = /trident.*rv\:11\./.test(ua),
60         isEdge = ua.indexOf("edge") > -1,
61         isGecko = !isSafari && ua.indexOf("gecko") > -1,
62         isBorderBox = isIE && !isStrict,
63         isWindows = (ua.indexOf("windows") != -1 || ua.indexOf("win32") != -1),
64         isMac = (ua.indexOf("macintosh") != -1 || ua.indexOf("mac os x") != -1),
65         isLinux = (ua.indexOf("linux") != -1),
66         isSecure = window.location.href.toLowerCase().indexOf("https") === 0,
67         isIOS = /iphone|ipad/.test(ua),
68         isAndroid = /android/.test(ua),
69         isTouch =  (function() {
70             try {
71                 if (ua.indexOf('chrome') != -1 && ua.indexOf('android') == -1) {
72                     window.addEventListener('touchstart', function __set_has_touch__ () {
73                         Roo.isTouch = true;
74                         window.removeEventListener('touchstart', __set_has_touch__);
75                     });
76                     return false; // no touch on chrome!?
77                 }
78                 document.createEvent("TouchEvent");  
79                 return true;  
80             } catch (e) {  
81                 return false;  
82             } 
83             
84         })();
85     // remove css image flicker
86         if(isIE && !isIE7){
87         try{
88             document.execCommand("BackgroundImageCache", false, true);
89         }catch(e){}
90     }
91     
92     Roo.apply(Roo, {
93         /**
94          * True if the browser is in strict mode
95          * @type Boolean
96          */
97         isStrict : isStrict,
98         /**
99          * True if the page is running over SSL
100          * @type Boolean
101          */
102         isSecure : isSecure,
103         /**
104          * True when the document is fully initialized and ready for action
105          * @type Boolean
106          */
107         isReady : false,
108         /**
109          * Turn on debugging output (currently only the factory uses this)
110          * @type Boolean
111          */
112         
113         debug: false,
114
115         /**
116          * True to automatically uncache orphaned Roo.Elements periodically (defaults to true)
117          * @type Boolean
118          */
119         enableGarbageCollector : true,
120
121         /**
122          * True to automatically purge event listeners after uncaching an element (defaults to false).
123          * Note: this only happens if enableGarbageCollector is true.
124          * @type Boolean
125          */
126         enableListenerCollection:false,
127
128         /**
129          * URL to a blank file used by Roo when in secure mode for iframe src and onReady src to prevent
130          * the IE insecure content warning (defaults to javascript:false).
131          * @type String
132          */
133         SSL_SECURE_URL : "javascript:false",
134
135         /**
136          * URL to a 1x1 transparent gif image used by Roo to create inline icons with CSS background images. (Defaults to
137          * "http://Roojs.com/s.gif" and you should change this to a URL on your server).
138          * @type String
139          */
140         BLANK_IMAGE_URL : "http:/"+"/localhost/s.gif",
141
142         emptyFn : function(){},
143         
144         /**
145          * Copies all the properties of config to obj if they don't already exist.
146          * @param {Object} obj The receiver of the properties
147          * @param {Object} config The source of the properties
148          * @return {Object} returns obj
149          */
150         applyIf : function(o, c){
151             if(o && c){
152                 for(var p in c){
153                     if(typeof o[p] == "undefined"){ o[p] = c[p]; }
154                 }
155             }
156             return o;
157         },
158
159         /**
160          * Applies event listeners to elements by selectors when the document is ready.
161          * The event name is specified with an @ suffix.
162 <pre><code>
163 Roo.addBehaviors({
164    // add a listener for click on all anchors in element with id foo
165    '#foo a@click' : function(e, t){
166        // do something
167    },
168
169    // add the same listener to multiple selectors (separated by comma BEFORE the @)
170    '#foo a, #bar span.some-class@mouseover' : function(){
171        // do something
172    }
173 });
174 </code></pre>
175          * @param {Object} obj The list of behaviors to apply
176          */
177         addBehaviors : function(o){
178             if(!Roo.isReady){
179                 Roo.onReady(function(){
180                     Roo.addBehaviors(o);
181                 });
182                 return;
183             }
184             var cache = {}; // simple cache for applying multiple behaviors to same selector does query multiple times
185             for(var b in o){
186                 var parts = b.split('@');
187                 if(parts[1]){ // for Object prototype breakers
188                     var s = parts[0];
189                     if(!cache[s]){
190                         cache[s] = Roo.select(s);
191                     }
192                     cache[s].on(parts[1], o[b]);
193                 }
194             }
195             cache = null;
196         },
197
198         /**
199          * Generates unique ids. If the element already has an id, it is unchanged
200          * @param {String/HTMLElement/Element} el (optional) The element to generate an id for
201          * @param {String} prefix (optional) Id prefix (defaults "Roo-gen")
202          * @return {String} The generated Id.
203          */
204         id : function(el, prefix){
205             prefix = prefix || "roo-gen";
206             el = Roo.getDom(el);
207             var id = prefix + (++idSeed);
208             return el ? (el.id ? el.id : (el.id = id)) : id;
209         },
210          
211        
212         /**
213          * Extends one class with another class and optionally overrides members with the passed literal. This class
214          * also adds the function "override()" to the class that can be used to override
215          * members on an instance.
216          * @param {Object} subclass The class inheriting the functionality
217          * @param {Object} superclass The class being extended
218          * @param {Object} overrides (optional) A literal with members
219          * @method extend
220          */
221         extend : function(){
222             // inline overrides
223             var io = function(o){
224                 for(var m in o){
225                     this[m] = o[m];
226                 }
227             };
228             return function(sb, sp, overrides){
229                 if(typeof sp == 'object'){ // eg. prototype, rather than function constructor..
230                     overrides = sp;
231                     sp = sb;
232                     sb = function(){sp.apply(this, arguments);};
233                 }
234                 var F = function(){}, sbp, spp = sp.prototype;
235                 F.prototype = spp;
236                 sbp = sb.prototype = new F();
237                 sbp.constructor=sb;
238                 sb.superclass=spp;
239                 
240                 if(spp.constructor == Object.prototype.constructor){
241                     spp.constructor=sp;
242                    
243                 }
244                 
245                 sb.override = function(o){
246                     Roo.override(sb, o);
247                 };
248                 sbp.override = io;
249                 Roo.override(sb, overrides);
250                 return sb;
251             };
252         }(),
253
254         /**
255          * Adds a list of functions to the prototype of an existing class, overwriting any existing methods with the same name.
256          * Usage:<pre><code>
257 Roo.override(MyClass, {
258     newMethod1: function(){
259         // etc.
260     },
261     newMethod2: function(foo){
262         // etc.
263     }
264 });
265  </code></pre>
266          * @param {Object} origclass The class to override
267          * @param {Object} overrides The list of functions to add to origClass.  This should be specified as an object literal
268          * containing one or more methods.
269          * @method override
270          */
271         override : function(origclass, overrides){
272             if(overrides){
273                 var p = origclass.prototype;
274                 for(var method in overrides){
275                     p[method] = overrides[method];
276                 }
277             }
278         },
279         /**
280          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
281          * <pre><code>
282 Roo.namespace('Company', 'Company.data');
283 Company.Widget = function() { ... }
284 Company.data.CustomStore = function(config) { ... }
285 </code></pre>
286          * @param {String} namespace1
287          * @param {String} namespace2
288          * @param {String} etc
289          * @method namespace
290          */
291         namespace : function(){
292             var a=arguments, o=null, i, j, d, rt;
293             for (i=0; i<a.length; ++i) {
294                 d=a[i].split(".");
295                 rt = d[0];
296                 /** eval:var:o */
297                 eval('if (typeof ' + rt + ' == "undefined"){' + rt + ' = {};} o = ' + rt + ';');
298                 for (j=1; j<d.length; ++j) {
299                     o[d[j]]=o[d[j]] || {};
300                     o=o[d[j]];
301                 }
302             }
303         },
304         /**
305          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
306          * <pre><code>
307 Roo.factory({ xns: Roo.data, xtype : 'Store', .....});
308 Roo.factory(conf, Roo.data);
309 </code></pre>
310          * @param {String} classname
311          * @param {String} namespace (optional)
312          * @method factory
313          */
314          
315         factory : function(c, ns)
316         {
317             // no xtype, no ns or c.xns - or forced off by c.xns
318             if (!c.xtype   || (!ns && !c.xns) ||  (c.xns === false)) { // not enough info...
319                 return c;
320             }
321             ns = c.xns ? c.xns : ns; // if c.xns is set, then use that..
322             if (c.constructor == ns[c.xtype]) {// already created...
323                 return c;
324             }
325             if (ns[c.xtype]) {
326                 if (Roo.debug) { Roo.log("Roo.Factory(" + c.xtype + ")"); }
327                 var ret = new ns[c.xtype](c);
328                 ret.xns = false;
329                 return ret;
330             }
331             c.xns = false; // prevent recursion..
332             return c;
333         },
334          /**
335          * Logs to console if it can.
336          *
337          * @param {String|Object} string
338          * @method log
339          */
340         log : function(s)
341         {
342             if ((typeof(console) == 'undefined') || (typeof(console.log) == 'undefined')) {
343                 return; // alerT?
344             }
345             
346             console.log(s);
347         },
348         /**
349          * Takes an object and converts it to an encoded URL. e.g. Roo.urlEncode({foo: 1, bar: 2}); would return "foo=1&bar=2".  Optionally, property values can be arrays, instead of keys and the resulting string that's returned will contain a name/value pair for each array value.
350          * @param {Object} o
351          * @return {String}
352          */
353         urlEncode : function(o){
354             if(!o){
355                 return "";
356             }
357             var buf = [];
358             for(var key in o){
359                 var ov = o[key], k = Roo.encodeURIComponent(key);
360                 var type = typeof ov;
361                 if(type == 'undefined'){
362                     buf.push(k, "=&");
363                 }else if(type != "function" && type != "object"){
364                     buf.push(k, "=", Roo.encodeURIComponent(ov), "&");
365                 }else if(ov instanceof Array){
366                     if (ov.length) {
367                             for(var i = 0, len = ov.length; i < len; i++) {
368                                 buf.push(k, "=", Roo.encodeURIComponent(ov[i] === undefined ? '' : ov[i]), "&");
369                             }
370                         } else {
371                             buf.push(k, "=&");
372                         }
373                 }
374             }
375             buf.pop();
376             return buf.join("");
377         },
378          /**
379          * Safe version of encodeURIComponent
380          * @param {String} data 
381          * @return {String} 
382          */
383         
384         encodeURIComponent : function (data)
385         {
386             try {
387                 return encodeURIComponent(data);
388             } catch(e) {} // should be an uri encode error.
389             
390             if (data == '' || data == null){
391                return '';
392             }
393             // http://stackoverflow.com/questions/2596483/unicode-and-uri-encoding-decoding-and-escaping-in-javascript
394             function nibble_to_hex(nibble){
395                 var chars = '0123456789ABCDEF';
396                 return chars.charAt(nibble);
397             }
398             data = data.toString();
399             var buffer = '';
400             for(var i=0; i<data.length; i++){
401                 var c = data.charCodeAt(i);
402                 var bs = new Array();
403                 if (c > 0x10000){
404                         // 4 bytes
405                     bs[0] = 0xF0 | ((c & 0x1C0000) >>> 18);
406                     bs[1] = 0x80 | ((c & 0x3F000) >>> 12);
407                     bs[2] = 0x80 | ((c & 0xFC0) >>> 6);
408                     bs[3] = 0x80 | (c & 0x3F);
409                 }else if (c > 0x800){
410                          // 3 bytes
411                     bs[0] = 0xE0 | ((c & 0xF000) >>> 12);
412                     bs[1] = 0x80 | ((c & 0xFC0) >>> 6);
413                     bs[2] = 0x80 | (c & 0x3F);
414                 }else if (c > 0x80){
415                        // 2 bytes
416                     bs[0] = 0xC0 | ((c & 0x7C0) >>> 6);
417                     bs[1] = 0x80 | (c & 0x3F);
418                 }else{
419                         // 1 byte
420                     bs[0] = c;
421                 }
422                 for(var j=0; j<bs.length; j++){
423                     var b = bs[j];
424                     var hex = nibble_to_hex((b & 0xF0) >>> 4) 
425                             + nibble_to_hex(b &0x0F);
426                     buffer += '%'+hex;
427                }
428             }
429             return buffer;    
430              
431         },
432
433         /**
434          * Takes an encoded URL and and converts it to an object. e.g. Roo.urlDecode("foo=1&bar=2"); would return {foo: 1, bar: 2} or Roo.urlDecode("foo=1&bar=2&bar=3&bar=4", true); would return {foo: 1, bar: [2, 3, 4]}.
435          * @param {String} string
436          * @param {Boolean} overwrite (optional) Items of the same name will overwrite previous values instead of creating an an array (Defaults to false).
437          * @return {Object} A literal with members
438          */
439         urlDecode : function(string, overwrite){
440             if(!string || !string.length){
441                 return {};
442             }
443             var obj = {};
444             var pairs = string.split('&');
445             var pair, name, value;
446             for(var i = 0, len = pairs.length; i < len; i++){
447                 pair = pairs[i].split('=');
448                 name = decodeURIComponent(pair[0]);
449                 value = decodeURIComponent(pair[1]);
450                 if(overwrite !== true){
451                     if(typeof obj[name] == "undefined"){
452                         obj[name] = value;
453                     }else if(typeof obj[name] == "string"){
454                         obj[name] = [obj[name]];
455                         obj[name].push(value);
456                     }else{
457                         obj[name].push(value);
458                     }
459                 }else{
460                     obj[name] = value;
461                 }
462             }
463             return obj;
464         },
465
466         /**
467          * Iterates an array calling the passed function with each item, stopping if your function returns false. If the
468          * passed array is not really an array, your function is called once with it.
469          * The supplied function is called with (Object item, Number index, Array allItems).
470          * @param {Array/NodeList/Mixed} array
471          * @param {Function} fn
472          * @param {Object} scope
473          */
474         each : function(array, fn, scope){
475             if(typeof array.length == "undefined" || typeof array == "string"){
476                 array = [array];
477             }
478             for(var i = 0, len = array.length; i < len; i++){
479                 if(fn.call(scope || array[i], array[i], i, array) === false){ return i; };
480             }
481         },
482
483         // deprecated
484         combine : function(){
485             var as = arguments, l = as.length, r = [];
486             for(var i = 0; i < l; i++){
487                 var a = as[i];
488                 if(a instanceof Array){
489                     r = r.concat(a);
490                 }else if(a.length !== undefined && !a.substr){
491                     r = r.concat(Array.prototype.slice.call(a, 0));
492                 }else{
493                     r.push(a);
494                 }
495             }
496             return r;
497         },
498
499         /**
500          * Escapes the passed string for use in a regular expression
501          * @param {String} str
502          * @return {String}
503          */
504         escapeRe : function(s) {
505             return s.replace(/([.*+?^${}()|[\]\/\\])/g, "\\$1");
506         },
507
508         // internal
509         callback : function(cb, scope, args, delay){
510             if(typeof cb == "function"){
511                 if(delay){
512                     cb.defer(delay, scope, args || []);
513                 }else{
514                     cb.apply(scope, args || []);
515                 }
516             }
517         },
518
519         /**
520          * Return the dom node for the passed string (id), dom node, or Roo.Element
521          * @param {String/HTMLElement/Roo.Element} el
522          * @return HTMLElement
523          */
524         getDom : function(el){
525             if(!el){
526                 return null;
527             }
528             return el.dom ? el.dom : (typeof el == 'string' ? document.getElementById(el) : el);
529         },
530
531         /**
532         * Shorthand for {@link Roo.ComponentMgr#get}
533         * @param {String} id
534         * @return Roo.Component
535         */
536         getCmp : function(id){
537             return Roo.ComponentMgr.get(id);
538         },
539          
540         num : function(v, defaultValue){
541             if(typeof v != 'number'){
542                 return defaultValue;
543             }
544             return v;
545         },
546
547         destroy : function(){
548             for(var i = 0, a = arguments, len = a.length; i < len; i++) {
549                 var as = a[i];
550                 if(as){
551                     if(as.dom){
552                         as.removeAllListeners();
553                         as.remove();
554                         continue;
555                     }
556                     if(typeof as.purgeListeners == 'function'){
557                         as.purgeListeners();
558                     }
559                     if(typeof as.destroy == 'function'){
560                         as.destroy();
561                     }
562                 }
563             }
564         },
565
566         // inpired by a similar function in mootools library
567         /**
568          * Returns the type of object that is passed in. If the object passed in is null or undefined it
569          * return false otherwise it returns one of the following values:<ul>
570          * <li><b>string</b>: If the object passed is a string</li>
571          * <li><b>number</b>: If the object passed is a number</li>
572          * <li><b>boolean</b>: If the object passed is a boolean value</li>
573          * <li><b>function</b>: If the object passed is a function reference</li>
574          * <li><b>object</b>: If the object passed is an object</li>
575          * <li><b>array</b>: If the object passed is an array</li>
576          * <li><b>regexp</b>: If the object passed is a regular expression</li>
577          * <li><b>element</b>: If the object passed is a DOM Element</li>
578          * <li><b>nodelist</b>: If the object passed is a DOM NodeList</li>
579          * <li><b>textnode</b>: If the object passed is a DOM text node and contains something other than whitespace</li>
580          * <li><b>whitespace</b>: If the object passed is a DOM text node and contains only whitespace</li>
581          * @param {Mixed} object
582          * @return {String}
583          */
584         type : function(o){
585             if(o === undefined || o === null){
586                 return false;
587             }
588             if(o.htmlElement){
589                 return 'element';
590             }
591             var t = typeof o;
592             if(t == 'object' && o.nodeName) {
593                 switch(o.nodeType) {
594                     case 1: return 'element';
595                     case 3: return (/\S/).test(o.nodeValue) ? 'textnode' : 'whitespace';
596                 }
597             }
598             if(t == 'object' || t == 'function') {
599                 switch(o.constructor) {
600                     case Array: return 'array';
601                     case RegExp: return 'regexp';
602                 }
603                 if(typeof o.length == 'number' && typeof o.item == 'function') {
604                     return 'nodelist';
605                 }
606             }
607             return t;
608         },
609
610         /**
611          * Returns true if the passed value is null, undefined or an empty string (optional).
612          * @param {Mixed} value The value to test
613          * @param {Boolean} allowBlank (optional) Pass true if an empty string is not considered empty
614          * @return {Boolean}
615          */
616         isEmpty : function(v, allowBlank){
617             return v === null || v === undefined || (!allowBlank ? v === '' : false);
618         },
619         
620         /** @type Boolean */
621         isOpera : isOpera,
622         /** @type Boolean */
623         isSafari : isSafari,
624         /** @type Boolean */
625         isFirefox : isFirefox,
626         /** @type Boolean */
627         isIE : isIE,
628         /** @type Boolean */
629         isIE7 : isIE7,
630         /** @type Boolean */
631         isIE11 : isIE11,
632         /** @type Boolean */
633         isEdge : isEdge,
634         /** @type Boolean */
635         isGecko : isGecko,
636         /** @type Boolean */
637         isBorderBox : isBorderBox,
638         /** @type Boolean */
639         isWindows : isWindows,
640         /** @type Boolean */
641         isLinux : isLinux,
642         /** @type Boolean */
643         isMac : isMac,
644         /** @type Boolean */
645         isIOS : isIOS,
646         /** @type Boolean */
647         isAndroid : isAndroid,
648         /** @type Boolean */
649         isTouch : isTouch,
650
651         /**
652          * By default, Ext intelligently decides whether floating elements should be shimmed. If you are using flash,
653          * you may want to set this to true.
654          * @type Boolean
655          */
656         useShims : ((isIE && !isIE7) || (isGecko && isMac)),
657         
658         
659                 
660         /**
661          * Selects a single element as a Roo Element
662          * This is about as close as you can get to jQuery's $('do crazy stuff')
663          * @param {String} selector The selector/xpath query
664          * @param {Node} root (optional) The start of the query (defaults to document).
665          * @return {Roo.Element}
666          */
667         selectNode : function(selector, root) 
668         {
669             var node = Roo.DomQuery.selectNode(selector,root);
670             return node ? Roo.get(node) : new Roo.Element(false);
671         }
672         
673     });
674
675
676 })();
677
678 Roo.namespace("Roo", "Roo.util", "Roo.grid", "Roo.dd", "Roo.tree", "Roo.data",
679                 "Roo.form", "Roo.menu", "Roo.state", "Roo.lib", "Roo.layout",
680                 "Roo.app", "Roo.ux",
681                 "Roo.bootstrap",
682                 "Roo.bootstrap.dash");
683 /*
684  * Based on:
685  * Ext JS Library 1.1.1
686  * Copyright(c) 2006-2007, Ext JS, LLC.
687  *
688  * Originally Released Under LGPL - original licence link has changed is not relivant.
689  *
690  * Fork - LGPL
691  * <script type="text/javascript">
692  */
693
694 (function() {    
695     // wrappedn so fnCleanup is not in global scope...
696     if(Roo.isIE) {
697         function fnCleanUp() {
698             var p = Function.prototype;
699             delete p.createSequence;
700             delete p.defer;
701             delete p.createDelegate;
702             delete p.createCallback;
703             delete p.createInterceptor;
704
705             window.detachEvent("onunload", fnCleanUp);
706         }
707         window.attachEvent("onunload", fnCleanUp);
708     }
709 })();
710
711
712 /**
713  * @class Function
714  * These functions are available on every Function object (any JavaScript function).
715  */
716 Roo.apply(Function.prototype, {
717      /**
718      * Creates a callback that passes arguments[0], arguments[1], arguments[2], ...
719      * Call directly on any function. Example: <code>myFunction.createCallback(myarg, myarg2)</code>
720      * Will create a function that is bound to those 2 args.
721      * @return {Function} The new function
722     */
723     createCallback : function(/*args...*/){
724         // make args available, in function below
725         var args = arguments;
726         var method = this;
727         return function() {
728             return method.apply(window, args);
729         };
730     },
731
732     /**
733      * Creates a delegate (callback) that sets the scope to obj.
734      * Call directly on any function. Example: <code>this.myFunction.createDelegate(this)</code>
735      * Will create a function that is automatically scoped to this.
736      * @param {Object} obj (optional) The object for which the scope is set
737      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
738      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
739      *                                             if a number the args are inserted at the specified position
740      * @return {Function} The new function
741      */
742     createDelegate : function(obj, args, appendArgs){
743         var method = this;
744         return function() {
745             var callArgs = args || arguments;
746             if(appendArgs === true){
747                 callArgs = Array.prototype.slice.call(arguments, 0);
748                 callArgs = callArgs.concat(args);
749             }else if(typeof appendArgs == "number"){
750                 callArgs = Array.prototype.slice.call(arguments, 0); // copy arguments first
751                 var applyArgs = [appendArgs, 0].concat(args); // create method call params
752                 Array.prototype.splice.apply(callArgs, applyArgs); // splice them in
753             }
754             return method.apply(obj || window, callArgs);
755         };
756     },
757
758     /**
759      * Calls this function after the number of millseconds specified.
760      * @param {Number} millis The number of milliseconds for the setTimeout call (if 0 the function is executed immediately)
761      * @param {Object} obj (optional) The object for which the scope is set
762      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
763      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
764      *                                             if a number the args are inserted at the specified position
765      * @return {Number} The timeout id that can be used with clearTimeout
766      */
767     defer : function(millis, obj, args, appendArgs){
768         var fn = this.createDelegate(obj, args, appendArgs);
769         if(millis){
770             return setTimeout(fn, millis);
771         }
772         fn();
773         return 0;
774     },
775     /**
776      * Create a combined function call sequence of the original function + the passed function.
777      * The resulting function returns the results of the original function.
778      * The passed fcn is called with the parameters of the original function
779      * @param {Function} fcn The function to sequence
780      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
781      * @return {Function} The new function
782      */
783     createSequence : function(fcn, scope){
784         if(typeof fcn != "function"){
785             return this;
786         }
787         var method = this;
788         return function() {
789             var retval = method.apply(this || window, arguments);
790             fcn.apply(scope || this || window, arguments);
791             return retval;
792         };
793     },
794
795     /**
796      * Creates an interceptor function. The passed fcn is called before the original one. If it returns false, the original one is not called.
797      * The resulting function returns the results of the original function.
798      * The passed fcn is called with the parameters of the original function.
799      * @addon
800      * @param {Function} fcn The function to call before the original
801      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
802      * @return {Function} The new function
803      */
804     createInterceptor : function(fcn, scope){
805         if(typeof fcn != "function"){
806             return this;
807         }
808         var method = this;
809         return function() {
810             fcn.target = this;
811             fcn.method = method;
812             if(fcn.apply(scope || this || window, arguments) === false){
813                 return;
814             }
815             return method.apply(this || window, arguments);
816         };
817     }
818 });
819 /*
820  * Based on:
821  * Ext JS Library 1.1.1
822  * Copyright(c) 2006-2007, Ext JS, LLC.
823  *
824  * Originally Released Under LGPL - original licence link has changed is not relivant.
825  *
826  * Fork - LGPL
827  * <script type="text/javascript">
828  */
829
830 Roo.applyIf(String, {
831     
832     /** @scope String */
833     
834     /**
835      * Escapes the passed string for ' and \
836      * @param {String} string The string to escape
837      * @return {String} The escaped string
838      * @static
839      */
840     escape : function(string) {
841         return string.replace(/('|\\)/g, "\\$1");
842     },
843
844     /**
845      * Pads the left side of a string with a specified character.  This is especially useful
846      * for normalizing number and date strings.  Example usage:
847      * <pre><code>
848 var s = String.leftPad('123', 5, '0');
849 // s now contains the string: '00123'
850 </code></pre>
851      * @param {String} string The original string
852      * @param {Number} size The total length of the output string
853      * @param {String} char (optional) The character with which to pad the original string (defaults to empty string " ")
854      * @return {String} The padded string
855      * @static
856      */
857     leftPad : function (val, size, ch) {
858         var result = new String(val);
859         if(ch === null || ch === undefined || ch === '') {
860             ch = " ";
861         }
862         while (result.length < size) {
863             result = ch + result;
864         }
865         return result;
866     },
867
868     /**
869      * Allows you to define a tokenized string and pass an arbitrary number of arguments to replace the tokens.  Each
870      * token must be unique, and must increment in the format {0}, {1}, etc.  Example usage:
871      * <pre><code>
872 var cls = 'my-class', text = 'Some text';
873 var s = String.format('<div class="{0}">{1}</div>', cls, text);
874 // s now contains the string: '<div class="my-class">Some text</div>'
875 </code></pre>
876      * @param {String} string The tokenized string to be formatted
877      * @param {String} value1 The value to replace token {0}
878      * @param {String} value2 Etc...
879      * @return {String} The formatted string
880      * @static
881      */
882     format : function(format){
883         var args = Array.prototype.slice.call(arguments, 1);
884         return format.replace(/\{(\d+)\}/g, function(m, i){
885             return Roo.util.Format.htmlEncode(args[i]);
886         });
887     }
888   
889     
890 });
891
892 /**
893  * Utility function that allows you to easily switch a string between two alternating values.  The passed value
894  * is compared to the current string, and if they are equal, the other value that was passed in is returned.  If
895  * they are already different, the first value passed in is returned.  Note that this method returns the new value
896  * but does not change the current string.
897  * <pre><code>
898 // alternate sort directions
899 sort = sort.toggle('ASC', 'DESC');
900
901 // instead of conditional logic:
902 sort = (sort == 'ASC' ? 'DESC' : 'ASC');
903 </code></pre>
904  * @param {String} value The value to compare to the current string
905  * @param {String} other The new value to use if the string already equals the first value passed in
906  * @return {String} The new value
907  */
908  
909 String.prototype.toggle = function(value, other){
910     return this == value ? other : value;
911 };
912
913
914 /**
915   * Remove invalid unicode characters from a string 
916   *
917   * @return {String} The clean string
918   */
919 String.prototype.unicodeClean = function () {
920     return this.replace(/[\s\S]/g,
921         function(character) {
922             if (character.charCodeAt()< 256) {
923               return character;
924            }
925            try {
926                 encodeURIComponent(character);
927            } catch(e) { 
928               return '';
929            }
930            return character;
931         }
932     );
933 };
934   
935 /*
936  * Based on:
937  * Ext JS Library 1.1.1
938  * Copyright(c) 2006-2007, Ext JS, LLC.
939  *
940  * Originally Released Under LGPL - original licence link has changed is not relivant.
941  *
942  * Fork - LGPL
943  * <script type="text/javascript">
944  */
945
946  /**
947  * @class Number
948  */
949 Roo.applyIf(Number.prototype, {
950     /**
951      * Checks whether or not the current number is within a desired range.  If the number is already within the
952      * range it is returned, otherwise the min or max value is returned depending on which side of the range is
953      * exceeded.  Note that this method returns the constrained value but does not change the current number.
954      * @param {Number} min The minimum number in the range
955      * @param {Number} max The maximum number in the range
956      * @return {Number} The constrained value if outside the range, otherwise the current value
957      */
958     constrain : function(min, max){
959         return Math.min(Math.max(this, min), max);
960     }
961 });/*
962  * Based on:
963  * Ext JS Library 1.1.1
964  * Copyright(c) 2006-2007, Ext JS, LLC.
965  *
966  * Originally Released Under LGPL - original licence link has changed is not relivant.
967  *
968  * Fork - LGPL
969  * <script type="text/javascript">
970  */
971  /**
972  * @class Array
973  */
974 Roo.applyIf(Array.prototype, {
975     /**
976      * 
977      * Checks whether or not the specified object exists in the array.
978      * @param {Object} o The object to check for
979      * @return {Number} The index of o in the array (or -1 if it is not found)
980      */
981     indexOf : function(o){
982        for (var i = 0, len = this.length; i < len; i++){
983               if(this[i] == o) { return i; }
984        }
985            return -1;
986     },
987
988     /**
989      * Removes the specified object from the array.  If the object is not found nothing happens.
990      * @param {Object} o The object to remove
991      */
992     remove : function(o){
993        var index = this.indexOf(o);
994        if(index != -1){
995            this.splice(index, 1);
996        }
997     },
998     /**
999      * Map (JS 1.6 compatibility)
1000      * @param {Function} function  to call
1001      */
1002     map : function(fun )
1003     {
1004         var len = this.length >>> 0;
1005         if (typeof fun != "function") {
1006             throw new TypeError();
1007         }
1008         var res = new Array(len);
1009         var thisp = arguments[1];
1010         for (var i = 0; i < len; i++)
1011         {
1012             if (i in this) {
1013                 res[i] = fun.call(thisp, this[i], i, this);
1014             }
1015         }
1016
1017         return res;
1018     }
1019     
1020 });
1021
1022
1023  
1024 /*
1025  * Based on:
1026  * Ext JS Library 1.1.1
1027  * Copyright(c) 2006-2007, Ext JS, LLC.
1028  *
1029  * Originally Released Under LGPL - original licence link has changed is not relivant.
1030  *
1031  * Fork - LGPL
1032  * <script type="text/javascript">
1033  */
1034
1035 /**
1036  * @class Date
1037  *
1038  * The date parsing and format syntax is a subset of
1039  * <a href="http://www.php.net/date">PHP's date() function</a>, and the formats that are
1040  * supported will provide results equivalent to their PHP versions.
1041  *
1042  * Following is the list of all currently supported formats:
1043  *<pre>
1044 Sample date:
1045 'Wed Jan 10 2007 15:05:01 GMT-0600 (Central Standard Time)'
1046
1047 Format  Output      Description
1048 ------  ----------  --------------------------------------------------------------
1049   d      10         Day of the month, 2 digits with leading zeros
1050   D      Wed        A textual representation of a day, three letters
1051   j      10         Day of the month without leading zeros
1052   l      Wednesday  A full textual representation of the day of the week
1053   S      th         English ordinal day of month suffix, 2 chars (use with j)
1054   w      3          Numeric representation of the day of the week
1055   z      9          The julian date, or day of the year (0-365)
1056   W      01         ISO-8601 2-digit week number of year, weeks starting on Monday (00-52)
1057   F      January    A full textual representation of the month
1058   m      01         Numeric representation of a month, with leading zeros
1059   M      Jan        Month name abbreviation, three letters
1060   n      1          Numeric representation of a month, without leading zeros
1061   t      31         Number of days in the given month
1062   L      0          Whether it's a leap year (1 if it is a leap year, else 0)
1063   Y      2007       A full numeric representation of a year, 4 digits
1064   y      07         A two digit representation of a year
1065   a      pm         Lowercase Ante meridiem and Post meridiem
1066   A      PM         Uppercase Ante meridiem and Post meridiem
1067   g      3          12-hour format of an hour without leading zeros
1068   G      15         24-hour format of an hour without leading zeros
1069   h      03         12-hour format of an hour with leading zeros
1070   H      15         24-hour format of an hour with leading zeros
1071   i      05         Minutes with leading zeros
1072   s      01         Seconds, with leading zeros
1073   O      -0600      Difference to Greenwich time (GMT) in hours (Allows +08, without minutes)
1074   P      -06:00     Difference to Greenwich time (GMT) with colon between hours and minutes
1075   T      CST        Timezone setting of the machine running the code
1076   Z      -21600     Timezone offset in seconds (negative if west of UTC, positive if east)
1077 </pre>
1078  *
1079  * Example usage (note that you must escape format specifiers with '\\' to render them as character literals):
1080  * <pre><code>
1081 var dt = new Date('1/10/2007 03:05:01 PM GMT-0600');
1082 document.write(dt.format('Y-m-d'));                         //2007-01-10
1083 document.write(dt.format('F j, Y, g:i a'));                 //January 10, 2007, 3:05 pm
1084 document.write(dt.format('l, \\t\\he dS of F Y h:i:s A'));  //Wednesday, the 10th of January 2007 03:05:01 PM
1085  </code></pre>
1086  *
1087  * Here are some standard date/time patterns that you might find helpful.  They
1088  * are not part of the source of Date.js, but to use them you can simply copy this
1089  * block of code into any script that is included after Date.js and they will also become
1090  * globally available on the Date object.  Feel free to add or remove patterns as needed in your code.
1091  * <pre><code>
1092 Date.patterns = {
1093     ISO8601Long:"Y-m-d H:i:s",
1094     ISO8601Short:"Y-m-d",
1095     ShortDate: "n/j/Y",
1096     LongDate: "l, F d, Y",
1097     FullDateTime: "l, F d, Y g:i:s A",
1098     MonthDay: "F d",
1099     ShortTime: "g:i A",
1100     LongTime: "g:i:s A",
1101     SortableDateTime: "Y-m-d\\TH:i:s",
1102     UniversalSortableDateTime: "Y-m-d H:i:sO",
1103     YearMonth: "F, Y"
1104 };
1105 </code></pre>
1106  *
1107  * Example usage:
1108  * <pre><code>
1109 var dt = new Date();
1110 document.write(dt.format(Date.patterns.ShortDate));
1111  </code></pre>
1112  */
1113
1114 /*
1115  * Most of the date-formatting functions below are the excellent work of Baron Schwartz.
1116  * They generate precompiled functions from date formats instead of parsing and
1117  * processing the pattern every time you format a date.  These functions are available
1118  * on every Date object (any javascript function).
1119  *
1120  * The original article and download are here:
1121  * http://www.xaprb.com/blog/2005/12/12/javascript-closures-for-runtime-efficiency/
1122  *
1123  */
1124  
1125  
1126  // was in core
1127 /**
1128  Returns the number of milliseconds between this date and date
1129  @param {Date} date (optional) Defaults to now
1130  @return {Number} The diff in milliseconds
1131  @member Date getElapsed
1132  */
1133 Date.prototype.getElapsed = function(date) {
1134         return Math.abs((date || new Date()).getTime()-this.getTime());
1135 };
1136 // was in date file..
1137
1138
1139 // private
1140 Date.parseFunctions = {count:0};
1141 // private
1142 Date.parseRegexes = [];
1143 // private
1144 Date.formatFunctions = {count:0};
1145
1146 // private
1147 Date.prototype.dateFormat = function(format) {
1148     if (Date.formatFunctions[format] == null) {
1149         Date.createNewFormat(format);
1150     }
1151     var func = Date.formatFunctions[format];
1152     return this[func]();
1153 };
1154
1155
1156 /**
1157  * Formats a date given the supplied format string
1158  * @param {String} format The format string
1159  * @return {String} The formatted date
1160  * @method
1161  */
1162 Date.prototype.format = Date.prototype.dateFormat;
1163
1164 // private
1165 Date.createNewFormat = function(format) {
1166     var funcName = "format" + Date.formatFunctions.count++;
1167     Date.formatFunctions[format] = funcName;
1168     var code = "Date.prototype." + funcName + " = function(){return ";
1169     var special = false;
1170     var ch = '';
1171     for (var i = 0; i < format.length; ++i) {
1172         ch = format.charAt(i);
1173         if (!special && ch == "\\") {
1174             special = true;
1175         }
1176         else if (special) {
1177             special = false;
1178             code += "'" + String.escape(ch) + "' + ";
1179         }
1180         else {
1181             code += Date.getFormatCode(ch);
1182         }
1183     }
1184     /** eval:var:zzzzzzzzzzzzz */
1185     eval(code.substring(0, code.length - 3) + ";}");
1186 };
1187
1188 // private
1189 Date.getFormatCode = function(character) {
1190     switch (character) {
1191     case "d":
1192         return "String.leftPad(this.getDate(), 2, '0') + ";
1193     case "D":
1194         return "Date.dayNames[this.getDay()].substring(0, 3) + ";
1195     case "j":
1196         return "this.getDate() + ";
1197     case "l":
1198         return "Date.dayNames[this.getDay()] + ";
1199     case "S":
1200         return "this.getSuffix() + ";
1201     case "w":
1202         return "this.getDay() + ";
1203     case "z":
1204         return "this.getDayOfYear() + ";
1205     case "W":
1206         return "this.getWeekOfYear() + ";
1207     case "F":
1208         return "Date.monthNames[this.getMonth()] + ";
1209     case "m":
1210         return "String.leftPad(this.getMonth() + 1, 2, '0') + ";
1211     case "M":
1212         return "Date.monthNames[this.getMonth()].substring(0, 3) + ";
1213     case "n":
1214         return "(this.getMonth() + 1) + ";
1215     case "t":
1216         return "this.getDaysInMonth() + ";
1217     case "L":
1218         return "(this.isLeapYear() ? 1 : 0) + ";
1219     case "Y":
1220         return "this.getFullYear() + ";
1221     case "y":
1222         return "('' + this.getFullYear()).substring(2, 4) + ";
1223     case "a":
1224         return "(this.getHours() < 12 ? 'am' : 'pm') + ";
1225     case "A":
1226         return "(this.getHours() < 12 ? 'AM' : 'PM') + ";
1227     case "g":
1228         return "((this.getHours() % 12) ? this.getHours() % 12 : 12) + ";
1229     case "G":
1230         return "this.getHours() + ";
1231     case "h":
1232         return "String.leftPad((this.getHours() % 12) ? this.getHours() % 12 : 12, 2, '0') + ";
1233     case "H":
1234         return "String.leftPad(this.getHours(), 2, '0') + ";
1235     case "i":
1236         return "String.leftPad(this.getMinutes(), 2, '0') + ";
1237     case "s":
1238         return "String.leftPad(this.getSeconds(), 2, '0') + ";
1239     case "O":
1240         return "this.getGMTOffset() + ";
1241     case "P":
1242         return "this.getGMTColonOffset() + ";
1243     case "T":
1244         return "this.getTimezone() + ";
1245     case "Z":
1246         return "(this.getTimezoneOffset() * -60) + ";
1247     default:
1248         return "'" + String.escape(character) + "' + ";
1249     }
1250 };
1251
1252 /**
1253  * Parses the passed string using the specified format. Note that this function expects dates in normal calendar
1254  * format, meaning that months are 1-based (1 = January) and not zero-based like in JavaScript dates.  Any part of
1255  * the date format that is not specified will default to the current date value for that part.  Time parts can also
1256  * be specified, but default to 0.  Keep in mind that the input date string must precisely match the specified format
1257  * string or the parse operation will fail.
1258  * Example Usage:
1259 <pre><code>
1260 //dt = Fri May 25 2007 (current date)
1261 var dt = new Date();
1262
1263 //dt = Thu May 25 2006 (today's month/day in 2006)
1264 dt = Date.parseDate("2006", "Y");
1265
1266 //dt = Sun Jan 15 2006 (all date parts specified)
1267 dt = Date.parseDate("2006-1-15", "Y-m-d");
1268
1269 //dt = Sun Jan 15 2006 15:20:01 GMT-0600 (CST)
1270 dt = Date.parseDate("2006-1-15 3:20:01 PM", "Y-m-d h:i:s A" );
1271 </code></pre>
1272  * @param {String} input The unparsed date as a string
1273  * @param {String} format The format the date is in
1274  * @return {Date} The parsed date
1275  * @static
1276  */
1277 Date.parseDate = function(input, format) {
1278     if (Date.parseFunctions[format] == null) {
1279         Date.createParser(format);
1280     }
1281     var func = Date.parseFunctions[format];
1282     return Date[func](input);
1283 };
1284 /**
1285  * @private
1286  */
1287
1288 Date.createParser = function(format) {
1289     var funcName = "parse" + Date.parseFunctions.count++;
1290     var regexNum = Date.parseRegexes.length;
1291     var currentGroup = 1;
1292     Date.parseFunctions[format] = funcName;
1293
1294     var code = "Date." + funcName + " = function(input){\n"
1295         + "var y = -1, m = -1, d = -1, h = -1, i = -1, s = -1, o, z, v;\n"
1296         + "var d = new Date();\n"
1297         + "y = d.getFullYear();\n"
1298         + "m = d.getMonth();\n"
1299         + "d = d.getDate();\n"
1300         + "if (typeof(input) !== 'string') { input = input.toString(); }\n"
1301         + "var results = input.match(Date.parseRegexes[" + regexNum + "]);\n"
1302         + "if (results && results.length > 0) {";
1303     var regex = "";
1304
1305     var special = false;
1306     var ch = '';
1307     for (var i = 0; i < format.length; ++i) {
1308         ch = format.charAt(i);
1309         if (!special && ch == "\\") {
1310             special = true;
1311         }
1312         else if (special) {
1313             special = false;
1314             regex += String.escape(ch);
1315         }
1316         else {
1317             var obj = Date.formatCodeToRegex(ch, currentGroup);
1318             currentGroup += obj.g;
1319             regex += obj.s;
1320             if (obj.g && obj.c) {
1321                 code += obj.c;
1322             }
1323         }
1324     }
1325
1326     code += "if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0 && s >= 0)\n"
1327         + "{v = new Date(y, m, d, h, i, s);}\n"
1328         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0)\n"
1329         + "{v = new Date(y, m, d, h, i);}\n"
1330         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0)\n"
1331         + "{v = new Date(y, m, d, h);}\n"
1332         + "else if (y >= 0 && m >= 0 && d > 0)\n"
1333         + "{v = new Date(y, m, d);}\n"
1334         + "else if (y >= 0 && m >= 0)\n"
1335         + "{v = new Date(y, m);}\n"
1336         + "else if (y >= 0)\n"
1337         + "{v = new Date(y);}\n"
1338         + "}return (v && (z || o))?\n" // favour UTC offset over GMT offset
1339         + "    ((z)? v.add(Date.SECOND, (v.getTimezoneOffset() * 60) + (z*1)) :\n" // reset to UTC, then add offset
1340         + "        v.add(Date.HOUR, (v.getGMTOffset() / 100) + (o / -100))) : v\n" // reset to GMT, then add offset
1341         + ";}";
1342
1343     Date.parseRegexes[regexNum] = new RegExp("^" + regex + "$");
1344     /** eval:var:zzzzzzzzzzzzz */
1345     eval(code);
1346 };
1347
1348 // private
1349 Date.formatCodeToRegex = function(character, currentGroup) {
1350     switch (character) {
1351     case "D":
1352         return {g:0,
1353         c:null,
1354         s:"(?:Sun|Mon|Tue|Wed|Thu|Fri|Sat)"};
1355     case "j":
1356         return {g:1,
1357             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1358             s:"(\\d{1,2})"}; // day of month without leading zeroes
1359     case "d":
1360         return {g:1,
1361             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1362             s:"(\\d{2})"}; // day of month with leading zeroes
1363     case "l":
1364         return {g:0,
1365             c:null,
1366             s:"(?:" + Date.dayNames.join("|") + ")"};
1367     case "S":
1368         return {g:0,
1369             c:null,
1370             s:"(?:st|nd|rd|th)"};
1371     case "w":
1372         return {g:0,
1373             c:null,
1374             s:"\\d"};
1375     case "z":
1376         return {g:0,
1377             c:null,
1378             s:"(?:\\d{1,3})"};
1379     case "W":
1380         return {g:0,
1381             c:null,
1382             s:"(?:\\d{2})"};
1383     case "F":
1384         return {g:1,
1385             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "].substring(0, 3)], 10);\n",
1386             s:"(" + Date.monthNames.join("|") + ")"};
1387     case "M":
1388         return {g:1,
1389             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "]], 10);\n",
1390             s:"(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)"};
1391     case "n":
1392         return {g:1,
1393             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1394             s:"(\\d{1,2})"}; // Numeric representation of a month, without leading zeros
1395     case "m":
1396         return {g:1,
1397             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1398             s:"(\\d{2})"}; // Numeric representation of a month, with leading zeros
1399     case "t":
1400         return {g:0,
1401             c:null,
1402             s:"\\d{1,2}"};
1403     case "L":
1404         return {g:0,
1405             c:null,
1406             s:"(?:1|0)"};
1407     case "Y":
1408         return {g:1,
1409             c:"y = parseInt(results[" + currentGroup + "], 10);\n",
1410             s:"(\\d{4})"};
1411     case "y":
1412         return {g:1,
1413             c:"var ty = parseInt(results[" + currentGroup + "], 10);\n"
1414                 + "y = ty > Date.y2kYear ? 1900 + ty : 2000 + ty;\n",
1415             s:"(\\d{1,2})"};
1416     case "a":
1417         return {g:1,
1418             c:"if (results[" + currentGroup + "] == 'am') {\n"
1419                 + "if (h == 12) { h = 0; }\n"
1420                 + "} else { if (h < 12) { h += 12; }}",
1421             s:"(am|pm)"};
1422     case "A":
1423         return {g:1,
1424             c:"if (results[" + currentGroup + "] == 'AM') {\n"
1425                 + "if (h == 12) { h = 0; }\n"
1426                 + "} else { if (h < 12) { h += 12; }}",
1427             s:"(AM|PM)"};
1428     case "g":
1429     case "G":
1430         return {g:1,
1431             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1432             s:"(\\d{1,2})"}; // 12/24-hr format  format of an hour without leading zeroes
1433     case "h":
1434     case "H":
1435         return {g:1,
1436             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1437             s:"(\\d{2})"}; //  12/24-hr format  format of an hour with leading zeroes
1438     case "i":
1439         return {g:1,
1440             c:"i = parseInt(results[" + currentGroup + "], 10);\n",
1441             s:"(\\d{2})"};
1442     case "s":
1443         return {g:1,
1444             c:"s = parseInt(results[" + currentGroup + "], 10);\n",
1445             s:"(\\d{2})"};
1446     case "O":
1447         return {g:1,
1448             c:[
1449                 "o = results[", currentGroup, "];\n",
1450                 "var sn = o.substring(0,1);\n", // get + / - sign
1451                 "var hr = o.substring(1,3)*1 + Math.floor(o.substring(3,5) / 60);\n", // get hours (performs minutes-to-hour conversion also)
1452                 "var mn = o.substring(3,5) % 60;\n", // get minutes
1453                 "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n", // -12hrs <= GMT offset <= 14hrs
1454                 "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1455             ].join(""),
1456             s:"([+\-]\\d{2,4})"};
1457     
1458     
1459     case "P":
1460         return {g:1,
1461                 c:[
1462                    "o = results[", currentGroup, "];\n",
1463                    "var sn = o.substring(0,1);\n",
1464                    "var hr = o.substring(1,3)*1 + Math.floor(o.substring(4,6) / 60);\n",
1465                    "var mn = o.substring(4,6) % 60;\n",
1466                    "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n",
1467                         "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1468             ].join(""),
1469             s:"([+\-]\\d{4})"};
1470     case "T":
1471         return {g:0,
1472             c:null,
1473             s:"[A-Z]{1,4}"}; // timezone abbrev. may be between 1 - 4 chars
1474     case "Z":
1475         return {g:1,
1476             c:"z = results[" + currentGroup + "];\n" // -43200 <= UTC offset <= 50400
1477                   + "z = (-43200 <= z*1 && z*1 <= 50400)? z : null;\n",
1478             s:"([+\-]?\\d{1,5})"}; // leading '+' sign is optional for UTC offset
1479     default:
1480         return {g:0,
1481             c:null,
1482             s:String.escape(character)};
1483     }
1484 };
1485
1486 /**
1487  * Get the timezone abbreviation of the current date (equivalent to the format specifier 'T').
1488  * @return {String} The abbreviated timezone name (e.g. 'CST')
1489  */
1490 Date.prototype.getTimezone = function() {
1491     return this.toString().replace(/^.*? ([A-Z]{1,4})[\-+][0-9]{4} .*$/, "$1");
1492 };
1493
1494 /**
1495  * Get the offset from GMT of the current date (equivalent to the format specifier 'O').
1496  * @return {String} The 4-character offset string prefixed with + or - (e.g. '-0600')
1497  */
1498 Date.prototype.getGMTOffset = function() {
1499     return (this.getTimezoneOffset() > 0 ? "-" : "+")
1500         + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1501         + String.leftPad(this.getTimezoneOffset() % 60, 2, "0");
1502 };
1503
1504 /**
1505  * Get the offset from GMT of the current date (equivalent to the format specifier 'P').
1506  * @return {String} 2-characters representing hours and 2-characters representing minutes
1507  * seperated by a colon and prefixed with + or - (e.g. '-06:00')
1508  */
1509 Date.prototype.getGMTColonOffset = function() {
1510         return (this.getTimezoneOffset() > 0 ? "-" : "+")
1511                 + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1512                 + ":"
1513                 + String.leftPad(this.getTimezoneOffset() %60, 2, "0");
1514 }
1515
1516 /**
1517  * Get the numeric day number of the year, adjusted for leap year.
1518  * @return {Number} 0 through 364 (365 in leap years)
1519  */
1520 Date.prototype.getDayOfYear = function() {
1521     var num = 0;
1522     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1523     for (var i = 0; i < this.getMonth(); ++i) {
1524         num += Date.daysInMonth[i];
1525     }
1526     return num + this.getDate() - 1;
1527 };
1528
1529 /**
1530  * Get the string representation of the numeric week number of the year
1531  * (equivalent to the format specifier 'W').
1532  * @return {String} '00' through '52'
1533  */
1534 Date.prototype.getWeekOfYear = function() {
1535     // Skip to Thursday of this week
1536     var now = this.getDayOfYear() + (4 - this.getDay());
1537     // Find the first Thursday of the year
1538     var jan1 = new Date(this.getFullYear(), 0, 1);
1539     var then = (7 - jan1.getDay() + 4);
1540     return String.leftPad(((now - then) / 7) + 1, 2, "0");
1541 };
1542
1543 /**
1544  * Whether or not the current date is in a leap year.
1545  * @return {Boolean} True if the current date is in a leap year, else false
1546  */
1547 Date.prototype.isLeapYear = function() {
1548     var year = this.getFullYear();
1549     return ((year & 3) == 0 && (year % 100 || (year % 400 == 0 && year)));
1550 };
1551
1552 /**
1553  * Get the first day of the current month, adjusted for leap year.  The returned value
1554  * is the numeric day index within the week (0-6) which can be used in conjunction with
1555  * the {@link #monthNames} array to retrieve the textual day name.
1556  * Example:
1557  *<pre><code>
1558 var dt = new Date('1/10/2007');
1559 document.write(Date.dayNames[dt.getFirstDayOfMonth()]); //output: 'Monday'
1560 </code></pre>
1561  * @return {Number} The day number (0-6)
1562  */
1563 Date.prototype.getFirstDayOfMonth = function() {
1564     var day = (this.getDay() - (this.getDate() - 1)) % 7;
1565     return (day < 0) ? (day + 7) : day;
1566 };
1567
1568 /**
1569  * Get the last day of the current month, adjusted for leap year.  The returned value
1570  * is the numeric day index within the week (0-6) which can be used in conjunction with
1571  * the {@link #monthNames} array to retrieve the textual day name.
1572  * Example:
1573  *<pre><code>
1574 var dt = new Date('1/10/2007');
1575 document.write(Date.dayNames[dt.getLastDayOfMonth()]); //output: 'Wednesday'
1576 </code></pre>
1577  * @return {Number} The day number (0-6)
1578  */
1579 Date.prototype.getLastDayOfMonth = function() {
1580     var day = (this.getDay() + (Date.daysInMonth[this.getMonth()] - this.getDate())) % 7;
1581     return (day < 0) ? (day + 7) : day;
1582 };
1583
1584
1585 /**
1586  * Get the first date of this date's month
1587  * @return {Date}
1588  */
1589 Date.prototype.getFirstDateOfMonth = function() {
1590     return new Date(this.getFullYear(), this.getMonth(), 1);
1591 };
1592
1593 /**
1594  * Get the last date of this date's month
1595  * @return {Date}
1596  */
1597 Date.prototype.getLastDateOfMonth = function() {
1598     return new Date(this.getFullYear(), this.getMonth(), this.getDaysInMonth());
1599 };
1600 /**
1601  * Get the number of days in the current month, adjusted for leap year.
1602  * @return {Number} The number of days in the month
1603  */
1604 Date.prototype.getDaysInMonth = function() {
1605     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1606     return Date.daysInMonth[this.getMonth()];
1607 };
1608
1609 /**
1610  * Get the English ordinal suffix of the current day (equivalent to the format specifier 'S').
1611  * @return {String} 'st, 'nd', 'rd' or 'th'
1612  */
1613 Date.prototype.getSuffix = function() {
1614     switch (this.getDate()) {
1615         case 1:
1616         case 21:
1617         case 31:
1618             return "st";
1619         case 2:
1620         case 22:
1621             return "nd";
1622         case 3:
1623         case 23:
1624             return "rd";
1625         default:
1626             return "th";
1627     }
1628 };
1629
1630 // private
1631 Date.daysInMonth = [31,28,31,30,31,30,31,31,30,31,30,31];
1632
1633 /**
1634  * An array of textual month names.
1635  * Override these values for international dates, for example...
1636  * Date.monthNames = ['JanInYourLang', 'FebInYourLang', ...];
1637  * @type Array
1638  * @static
1639  */
1640 Date.monthNames =
1641    ["January",
1642     "February",
1643     "March",
1644     "April",
1645     "May",
1646     "June",
1647     "July",
1648     "August",
1649     "September",
1650     "October",
1651     "November",
1652     "December"];
1653
1654 /**
1655  * An array of textual day names.
1656  * Override these values for international dates, for example...
1657  * Date.dayNames = ['SundayInYourLang', 'MondayInYourLang', ...];
1658  * @type Array
1659  * @static
1660  */
1661 Date.dayNames =
1662    ["Sunday",
1663     "Monday",
1664     "Tuesday",
1665     "Wednesday",
1666     "Thursday",
1667     "Friday",
1668     "Saturday"];
1669
1670 // private
1671 Date.y2kYear = 50;
1672 // private
1673 Date.monthNumbers = {
1674     Jan:0,
1675     Feb:1,
1676     Mar:2,
1677     Apr:3,
1678     May:4,
1679     Jun:5,
1680     Jul:6,
1681     Aug:7,
1682     Sep:8,
1683     Oct:9,
1684     Nov:10,
1685     Dec:11};
1686
1687 /**
1688  * Creates and returns a new Date instance with the exact same date value as the called instance.
1689  * Dates are copied and passed by reference, so if a copied date variable is modified later, the original
1690  * variable will also be changed.  When the intention is to create a new variable that will not
1691  * modify the original instance, you should create a clone.
1692  *
1693  * Example of correctly cloning a date:
1694  * <pre><code>
1695 //wrong way:
1696 var orig = new Date('10/1/2006');
1697 var copy = orig;
1698 copy.setDate(5);
1699 document.write(orig);  //returns 'Thu Oct 05 2006'!
1700
1701 //correct way:
1702 var orig = new Date('10/1/2006');
1703 var copy = orig.clone();
1704 copy.setDate(5);
1705 document.write(orig);  //returns 'Thu Oct 01 2006'
1706 </code></pre>
1707  * @return {Date} The new Date instance
1708  */
1709 Date.prototype.clone = function() {
1710         return new Date(this.getTime());
1711 };
1712
1713 /**
1714  * Clears any time information from this date
1715  @param {Boolean} clone true to create a clone of this date, clear the time and return it
1716  @return {Date} this or the clone
1717  */
1718 Date.prototype.clearTime = function(clone){
1719     if(clone){
1720         return this.clone().clearTime();
1721     }
1722     this.setHours(0);
1723     this.setMinutes(0);
1724     this.setSeconds(0);
1725     this.setMilliseconds(0);
1726     return this;
1727 };
1728
1729 // private
1730 // safari setMonth is broken -- check that this is only donw once...
1731 if(Roo.isSafari && typeof(Date.brokenSetMonth) == 'undefined'){
1732     Date.brokenSetMonth = Date.prototype.setMonth;
1733         Date.prototype.setMonth = function(num){
1734                 if(num <= -1){
1735                         var n = Math.ceil(-num);
1736                         var back_year = Math.ceil(n/12);
1737                         var month = (n % 12) ? 12 - n % 12 : 0 ;
1738                         this.setFullYear(this.getFullYear() - back_year);
1739                         return Date.brokenSetMonth.call(this, month);
1740                 } else {
1741                         return Date.brokenSetMonth.apply(this, arguments);
1742                 }
1743         };
1744 }
1745
1746 /** Date interval constant 
1747 * @static 
1748 * @type String */
1749 Date.MILLI = "ms";
1750 /** Date interval constant 
1751 * @static 
1752 * @type String */
1753 Date.SECOND = "s";
1754 /** Date interval constant 
1755 * @static 
1756 * @type String */
1757 Date.MINUTE = "mi";
1758 /** Date interval constant 
1759 * @static 
1760 * @type String */
1761 Date.HOUR = "h";
1762 /** Date interval constant 
1763 * @static 
1764 * @type String */
1765 Date.DAY = "d";
1766 /** Date interval constant 
1767 * @static 
1768 * @type String */
1769 Date.MONTH = "mo";
1770 /** Date interval constant 
1771 * @static 
1772 * @type String */
1773 Date.YEAR = "y";
1774
1775 /**
1776  * Provides a convenient method of performing basic date arithmetic.  This method
1777  * does not modify the Date instance being called - it creates and returns
1778  * a new Date instance containing the resulting date value.
1779  *
1780  * Examples:
1781  * <pre><code>
1782 //Basic usage:
1783 var dt = new Date('10/29/2006').add(Date.DAY, 5);
1784 document.write(dt); //returns 'Fri Oct 06 2006 00:00:00'
1785
1786 //Negative values will subtract correctly:
1787 var dt2 = new Date('10/1/2006').add(Date.DAY, -5);
1788 document.write(dt2); //returns 'Tue Sep 26 2006 00:00:00'
1789
1790 //You can even chain several calls together in one line!
1791 var dt3 = new Date('10/1/2006').add(Date.DAY, 5).add(Date.HOUR, 8).add(Date.MINUTE, -30);
1792 document.write(dt3); //returns 'Fri Oct 06 2006 07:30:00'
1793  </code></pre>
1794  *
1795  * @param {String} interval   A valid date interval enum value
1796  * @param {Number} value      The amount to add to the current date
1797  * @return {Date} The new Date instance
1798  */
1799 Date.prototype.add = function(interval, value){
1800   var d = this.clone();
1801   if (!interval || value === 0) { return d; }
1802   switch(interval.toLowerCase()){
1803     case Date.MILLI:
1804       d.setMilliseconds(this.getMilliseconds() + value);
1805       break;
1806     case Date.SECOND:
1807       d.setSeconds(this.getSeconds() + value);
1808       break;
1809     case Date.MINUTE:
1810       d.setMinutes(this.getMinutes() + value);
1811       break;
1812     case Date.HOUR:
1813       d.setHours(this.getHours() + value);
1814       break;
1815     case Date.DAY:
1816       d.setDate(this.getDate() + value);
1817       break;
1818     case Date.MONTH:
1819       var day = this.getDate();
1820       if(day > 28){
1821           day = Math.min(day, this.getFirstDateOfMonth().add('mo', value).getLastDateOfMonth().getDate());
1822       }
1823       d.setDate(day);
1824       d.setMonth(this.getMonth() + value);
1825       break;
1826     case Date.YEAR:
1827       d.setFullYear(this.getFullYear() + value);
1828       break;
1829   }
1830   return d;
1831 };
1832 /*
1833  * Based on:
1834  * Ext JS Library 1.1.1
1835  * Copyright(c) 2006-2007, Ext JS, LLC.
1836  *
1837  * Originally Released Under LGPL - original licence link has changed is not relivant.
1838  *
1839  * Fork - LGPL
1840  * <script type="text/javascript">
1841  */
1842
1843 /**
1844  * @class Roo.lib.Dom
1845  * @static
1846  * 
1847  * Dom utils (from YIU afaik)
1848  * 
1849  **/
1850 Roo.lib.Dom = {
1851     /**
1852      * Get the view width
1853      * @param {Boolean} full True will get the full document, otherwise it's the view width
1854      * @return {Number} The width
1855      */
1856      
1857     getViewWidth : function(full) {
1858         return full ? this.getDocumentWidth() : this.getViewportWidth();
1859     },
1860     /**
1861      * Get the view height
1862      * @param {Boolean} full True will get the full document, otherwise it's the view height
1863      * @return {Number} The height
1864      */
1865     getViewHeight : function(full) {
1866         return full ? this.getDocumentHeight() : this.getViewportHeight();
1867     },
1868
1869     getDocumentHeight: function() {
1870         var scrollHeight = (document.compatMode != "CSS1Compat") ? document.body.scrollHeight : document.documentElement.scrollHeight;
1871         return Math.max(scrollHeight, this.getViewportHeight());
1872     },
1873
1874     getDocumentWidth: function() {
1875         var scrollWidth = (document.compatMode != "CSS1Compat") ? document.body.scrollWidth : document.documentElement.scrollWidth;
1876         return Math.max(scrollWidth, this.getViewportWidth());
1877     },
1878
1879     getViewportHeight: function() {
1880         var height = self.innerHeight;
1881         var mode = document.compatMode;
1882
1883         if ((mode || Roo.isIE) && !Roo.isOpera) {
1884             height = (mode == "CSS1Compat") ?
1885                      document.documentElement.clientHeight :
1886                      document.body.clientHeight;
1887         }
1888
1889         return height;
1890     },
1891
1892     getViewportWidth: function() {
1893         var width = self.innerWidth;
1894         var mode = document.compatMode;
1895
1896         if (mode || Roo.isIE) {
1897             width = (mode == "CSS1Compat") ?
1898                     document.documentElement.clientWidth :
1899                     document.body.clientWidth;
1900         }
1901         return width;
1902     },
1903
1904     isAncestor : function(p, c) {
1905         p = Roo.getDom(p);
1906         c = Roo.getDom(c);
1907         if (!p || !c) {
1908             return false;
1909         }
1910
1911         if (p.contains && !Roo.isSafari) {
1912             return p.contains(c);
1913         } else if (p.compareDocumentPosition) {
1914             return !!(p.compareDocumentPosition(c) & 16);
1915         } else {
1916             var parent = c.parentNode;
1917             while (parent) {
1918                 if (parent == p) {
1919                     return true;
1920                 }
1921                 else if (!parent.tagName || parent.tagName.toUpperCase() == "HTML") {
1922                     return false;
1923                 }
1924                 parent = parent.parentNode;
1925             }
1926             return false;
1927         }
1928     },
1929
1930     getRegion : function(el) {
1931         return Roo.lib.Region.getRegion(el);
1932     },
1933
1934     getY : function(el) {
1935         return this.getXY(el)[1];
1936     },
1937
1938     getX : function(el) {
1939         return this.getXY(el)[0];
1940     },
1941
1942     getXY : function(el) {
1943         var p, pe, b, scroll, bd = document.body;
1944         el = Roo.getDom(el);
1945         var fly = Roo.lib.AnimBase.fly;
1946         if (el.getBoundingClientRect) {
1947             b = el.getBoundingClientRect();
1948             scroll = fly(document).getScroll();
1949             return [b.left + scroll.left, b.top + scroll.top];
1950         }
1951         var x = 0, y = 0;
1952
1953         p = el;
1954
1955         var hasAbsolute = fly(el).getStyle("position") == "absolute";
1956
1957         while (p) {
1958
1959             x += p.offsetLeft;
1960             y += p.offsetTop;
1961
1962             if (!hasAbsolute && fly(p).getStyle("position") == "absolute") {
1963                 hasAbsolute = true;
1964             }
1965
1966             if (Roo.isGecko) {
1967                 pe = fly(p);
1968
1969                 var bt = parseInt(pe.getStyle("borderTopWidth"), 10) || 0;
1970                 var bl = parseInt(pe.getStyle("borderLeftWidth"), 10) || 0;
1971
1972
1973                 x += bl;
1974                 y += bt;
1975
1976
1977                 if (p != el && pe.getStyle('overflow') != 'visible') {
1978                     x += bl;
1979                     y += bt;
1980                 }
1981             }
1982             p = p.offsetParent;
1983         }
1984
1985         if (Roo.isSafari && hasAbsolute) {
1986             x -= bd.offsetLeft;
1987             y -= bd.offsetTop;
1988         }
1989
1990         if (Roo.isGecko && !hasAbsolute) {
1991             var dbd = fly(bd);
1992             x += parseInt(dbd.getStyle("borderLeftWidth"), 10) || 0;
1993             y += parseInt(dbd.getStyle("borderTopWidth"), 10) || 0;
1994         }
1995
1996         p = el.parentNode;
1997         while (p && p != bd) {
1998             if (!Roo.isOpera || (p.tagName != 'TR' && fly(p).getStyle("display") != "inline")) {
1999                 x -= p.scrollLeft;
2000                 y -= p.scrollTop;
2001             }
2002             p = p.parentNode;
2003         }
2004         return [x, y];
2005     },
2006  
2007   
2008
2009
2010     setXY : function(el, xy) {
2011         el = Roo.fly(el, '_setXY');
2012         el.position();
2013         var pts = el.translatePoints(xy);
2014         if (xy[0] !== false) {
2015             el.dom.style.left = pts.left + "px";
2016         }
2017         if (xy[1] !== false) {
2018             el.dom.style.top = pts.top + "px";
2019         }
2020     },
2021
2022     setX : function(el, x) {
2023         this.setXY(el, [x, false]);
2024     },
2025
2026     setY : function(el, y) {
2027         this.setXY(el, [false, y]);
2028     }
2029 };
2030 /*
2031  * Portions of this file are based on pieces of Yahoo User Interface Library
2032  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2033  * YUI licensed under the BSD License:
2034  * http://developer.yahoo.net/yui/license.txt
2035  * <script type="text/javascript">
2036  *
2037  */
2038
2039 Roo.lib.Event = function() {
2040     var loadComplete = false;
2041     var listeners = [];
2042     var unloadListeners = [];
2043     var retryCount = 0;
2044     var onAvailStack = [];
2045     var counter = 0;
2046     var lastError = null;
2047
2048     return {
2049         POLL_RETRYS: 200,
2050         POLL_INTERVAL: 20,
2051         EL: 0,
2052         TYPE: 1,
2053         FN: 2,
2054         WFN: 3,
2055         OBJ: 3,
2056         ADJ_SCOPE: 4,
2057         _interval: null,
2058
2059         startInterval: function() {
2060             if (!this._interval) {
2061                 var self = this;
2062                 var callback = function() {
2063                     self._tryPreloadAttach();
2064                 };
2065                 this._interval = setInterval(callback, this.POLL_INTERVAL);
2066
2067             }
2068         },
2069
2070         onAvailable: function(p_id, p_fn, p_obj, p_override) {
2071             onAvailStack.push({ id:         p_id,
2072                 fn:         p_fn,
2073                 obj:        p_obj,
2074                 override:   p_override,
2075                 checkReady: false    });
2076
2077             retryCount = this.POLL_RETRYS;
2078             this.startInterval();
2079         },
2080
2081
2082         addListener: function(el, eventName, fn) {
2083             el = Roo.getDom(el);
2084             if (!el || !fn) {
2085                 return false;
2086             }
2087
2088             if ("unload" == eventName) {
2089                 unloadListeners[unloadListeners.length] =
2090                 [el, eventName, fn];
2091                 return true;
2092             }
2093
2094             var wrappedFn = function(e) {
2095                 return fn(Roo.lib.Event.getEvent(e));
2096             };
2097
2098             var li = [el, eventName, fn, wrappedFn];
2099
2100             var index = listeners.length;
2101             listeners[index] = li;
2102
2103             this.doAdd(el, eventName, wrappedFn, false);
2104             return true;
2105
2106         },
2107
2108
2109         removeListener: function(el, eventName, fn) {
2110             var i, len;
2111
2112             el = Roo.getDom(el);
2113
2114             if(!fn) {
2115                 return this.purgeElement(el, false, eventName);
2116             }
2117
2118
2119             if ("unload" == eventName) {
2120
2121                 for (i = 0,len = unloadListeners.length; i < len; i++) {
2122                     var li = unloadListeners[i];
2123                     if (li &&
2124                         li[0] == el &&
2125                         li[1] == eventName &&
2126                         li[2] == fn) {
2127                         unloadListeners.splice(i, 1);
2128                         return true;
2129                     }
2130                 }
2131
2132                 return false;
2133             }
2134
2135             var cacheItem = null;
2136
2137
2138             var index = arguments[3];
2139
2140             if ("undefined" == typeof index) {
2141                 index = this._getCacheIndex(el, eventName, fn);
2142             }
2143
2144             if (index >= 0) {
2145                 cacheItem = listeners[index];
2146             }
2147
2148             if (!el || !cacheItem) {
2149                 return false;
2150             }
2151
2152             this.doRemove(el, eventName, cacheItem[this.WFN], false);
2153
2154             delete listeners[index][this.WFN];
2155             delete listeners[index][this.FN];
2156             listeners.splice(index, 1);
2157
2158             return true;
2159
2160         },
2161
2162
2163         getTarget: function(ev, resolveTextNode) {
2164             ev = ev.browserEvent || ev;
2165             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2166             var t = ev.target || ev.srcElement;
2167             return this.resolveTextNode(t);
2168         },
2169
2170
2171         resolveTextNode: function(node) {
2172             if (Roo.isSafari && node && 3 == node.nodeType) {
2173                 return node.parentNode;
2174             } else {
2175                 return node;
2176             }
2177         },
2178
2179
2180         getPageX: function(ev) {
2181             ev = ev.browserEvent || ev;
2182             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2183             var x = ev.pageX;
2184             if (!x && 0 !== x) {
2185                 x = ev.clientX || 0;
2186
2187                 if (Roo.isIE) {
2188                     x += this.getScroll()[1];
2189                 }
2190             }
2191
2192             return x;
2193         },
2194
2195
2196         getPageY: function(ev) {
2197             ev = ev.browserEvent || ev;
2198             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2199             var y = ev.pageY;
2200             if (!y && 0 !== y) {
2201                 y = ev.clientY || 0;
2202
2203                 if (Roo.isIE) {
2204                     y += this.getScroll()[0];
2205                 }
2206             }
2207
2208
2209             return y;
2210         },
2211
2212
2213         getXY: function(ev) {
2214             ev = ev.browserEvent || ev;
2215             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2216             return [this.getPageX(ev), this.getPageY(ev)];
2217         },
2218
2219
2220         getRelatedTarget: function(ev) {
2221             ev = ev.browserEvent || ev;
2222             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2223             var t = ev.relatedTarget;
2224             if (!t) {
2225                 if (ev.type == "mouseout") {
2226                     t = ev.toElement;
2227                 } else if (ev.type == "mouseover") {
2228                     t = ev.fromElement;
2229                 }
2230             }
2231
2232             return this.resolveTextNode(t);
2233         },
2234
2235
2236         getTime: function(ev) {
2237             ev = ev.browserEvent || ev;
2238             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2239             if (!ev.time) {
2240                 var t = new Date().getTime();
2241                 try {
2242                     ev.time = t;
2243                 } catch(ex) {
2244                     this.lastError = ex;
2245                     return t;
2246                 }
2247             }
2248
2249             return ev.time;
2250         },
2251
2252
2253         stopEvent: function(ev) {
2254             this.stopPropagation(ev);
2255             this.preventDefault(ev);
2256         },
2257
2258
2259         stopPropagation: function(ev) {
2260             ev = ev.browserEvent || ev;
2261             if (ev.stopPropagation) {
2262                 ev.stopPropagation();
2263             } else {
2264                 ev.cancelBubble = true;
2265             }
2266         },
2267
2268
2269         preventDefault: function(ev) {
2270             ev = ev.browserEvent || ev;
2271             if(ev.preventDefault) {
2272                 ev.preventDefault();
2273             } else {
2274                 ev.returnValue = false;
2275             }
2276         },
2277
2278
2279         getEvent: function(e) {
2280             var ev = e || window.event;
2281             if (!ev) {
2282                 var c = this.getEvent.caller;
2283                 while (c) {
2284                     ev = c.arguments[0];
2285                     if (ev && Event == ev.constructor) {
2286                         break;
2287                     }
2288                     c = c.caller;
2289                 }
2290             }
2291             return ev;
2292         },
2293
2294
2295         getCharCode: function(ev) {
2296             ev = ev.browserEvent || ev;
2297             return ev.charCode || ev.keyCode || 0;
2298         },
2299
2300
2301         _getCacheIndex: function(el, eventName, fn) {
2302             for (var i = 0,len = listeners.length; i < len; ++i) {
2303                 var li = listeners[i];
2304                 if (li &&
2305                     li[this.FN] == fn &&
2306                     li[this.EL] == el &&
2307                     li[this.TYPE] == eventName) {
2308                     return i;
2309                 }
2310             }
2311
2312             return -1;
2313         },
2314
2315
2316         elCache: {},
2317
2318
2319         getEl: function(id) {
2320             return document.getElementById(id);
2321         },
2322
2323
2324         clearCache: function() {
2325         },
2326
2327
2328         _load: function(e) {
2329             loadComplete = true;
2330             var EU = Roo.lib.Event;
2331
2332
2333             if (Roo.isIE) {
2334                 EU.doRemove(window, "load", EU._load);
2335             }
2336         },
2337
2338
2339         _tryPreloadAttach: function() {
2340
2341             if (this.locked) {
2342                 return false;
2343             }
2344
2345             this.locked = true;
2346
2347
2348             var tryAgain = !loadComplete;
2349             if (!tryAgain) {
2350                 tryAgain = (retryCount > 0);
2351             }
2352
2353
2354             var notAvail = [];
2355             for (var i = 0,len = onAvailStack.length; i < len; ++i) {
2356                 var item = onAvailStack[i];
2357                 if (item) {
2358                     var el = this.getEl(item.id);
2359
2360                     if (el) {
2361                         if (!item.checkReady ||
2362                             loadComplete ||
2363                             el.nextSibling ||
2364                             (document && document.body)) {
2365
2366                             var scope = el;
2367                             if (item.override) {
2368                                 if (item.override === true) {
2369                                     scope = item.obj;
2370                                 } else {
2371                                     scope = item.override;
2372                                 }
2373                             }
2374                             item.fn.call(scope, item.obj);
2375                             onAvailStack[i] = null;
2376                         }
2377                     } else {
2378                         notAvail.push(item);
2379                     }
2380                 }
2381             }
2382
2383             retryCount = (notAvail.length === 0) ? 0 : retryCount - 1;
2384
2385             if (tryAgain) {
2386
2387                 this.startInterval();
2388             } else {
2389                 clearInterval(this._interval);
2390                 this._interval = null;
2391             }
2392
2393             this.locked = false;
2394
2395             return true;
2396
2397         },
2398
2399
2400         purgeElement: function(el, recurse, eventName) {
2401             var elListeners = this.getListeners(el, eventName);
2402             if (elListeners) {
2403                 for (var i = 0,len = elListeners.length; i < len; ++i) {
2404                     var l = elListeners[i];
2405                     this.removeListener(el, l.type, l.fn);
2406                 }
2407             }
2408
2409             if (recurse && el && el.childNodes) {
2410                 for (i = 0,len = el.childNodes.length; i < len; ++i) {
2411                     this.purgeElement(el.childNodes[i], recurse, eventName);
2412                 }
2413             }
2414         },
2415
2416
2417         getListeners: function(el, eventName) {
2418             var results = [], searchLists;
2419             if (!eventName) {
2420                 searchLists = [listeners, unloadListeners];
2421             } else if (eventName == "unload") {
2422                 searchLists = [unloadListeners];
2423             } else {
2424                 searchLists = [listeners];
2425             }
2426
2427             for (var j = 0; j < searchLists.length; ++j) {
2428                 var searchList = searchLists[j];
2429                 if (searchList && searchList.length > 0) {
2430                     for (var i = 0,len = searchList.length; i < len; ++i) {
2431                         var l = searchList[i];
2432                         if (l && l[this.EL] === el &&
2433                             (!eventName || eventName === l[this.TYPE])) {
2434                             results.push({
2435                                 type:   l[this.TYPE],
2436                                 fn:     l[this.FN],
2437                                 obj:    l[this.OBJ],
2438                                 adjust: l[this.ADJ_SCOPE],
2439                                 index:  i
2440                             });
2441                         }
2442                     }
2443                 }
2444             }
2445
2446             return (results.length) ? results : null;
2447         },
2448
2449
2450         _unload: function(e) {
2451
2452             var EU = Roo.lib.Event, i, j, l, len, index;
2453
2454             for (i = 0,len = unloadListeners.length; i < len; ++i) {
2455                 l = unloadListeners[i];
2456                 if (l) {
2457                     var scope = window;
2458                     if (l[EU.ADJ_SCOPE]) {
2459                         if (l[EU.ADJ_SCOPE] === true) {
2460                             scope = l[EU.OBJ];
2461                         } else {
2462                             scope = l[EU.ADJ_SCOPE];
2463                         }
2464                     }
2465                     l[EU.FN].call(scope, EU.getEvent(e), l[EU.OBJ]);
2466                     unloadListeners[i] = null;
2467                     l = null;
2468                     scope = null;
2469                 }
2470             }
2471
2472             unloadListeners = null;
2473
2474             if (listeners && listeners.length > 0) {
2475                 j = listeners.length;
2476                 while (j) {
2477                     index = j - 1;
2478                     l = listeners[index];
2479                     if (l) {
2480                         EU.removeListener(l[EU.EL], l[EU.TYPE],
2481                                 l[EU.FN], index);
2482                     }
2483                     j = j - 1;
2484                 }
2485                 l = null;
2486
2487                 EU.clearCache();
2488             }
2489
2490             EU.doRemove(window, "unload", EU._unload);
2491
2492         },
2493
2494
2495         getScroll: function() {
2496             var dd = document.documentElement, db = document.body;
2497             if (dd && (dd.scrollTop || dd.scrollLeft)) {
2498                 return [dd.scrollTop, dd.scrollLeft];
2499             } else if (db) {
2500                 return [db.scrollTop, db.scrollLeft];
2501             } else {
2502                 return [0, 0];
2503             }
2504         },
2505
2506
2507         doAdd: function () {
2508             if (window.addEventListener) {
2509                 return function(el, eventName, fn, capture) {
2510                     el.addEventListener(eventName, fn, (capture));
2511                 };
2512             } else if (window.attachEvent) {
2513                 return function(el, eventName, fn, capture) {
2514                     el.attachEvent("on" + eventName, fn);
2515                 };
2516             } else {
2517                 return function() {
2518                 };
2519             }
2520         }(),
2521
2522
2523         doRemove: function() {
2524             if (window.removeEventListener) {
2525                 return function (el, eventName, fn, capture) {
2526                     el.removeEventListener(eventName, fn, (capture));
2527                 };
2528             } else if (window.detachEvent) {
2529                 return function (el, eventName, fn) {
2530                     el.detachEvent("on" + eventName, fn);
2531                 };
2532             } else {
2533                 return function() {
2534                 };
2535             }
2536         }()
2537     };
2538     
2539 }();
2540 (function() {     
2541    
2542     var E = Roo.lib.Event;
2543     E.on = E.addListener;
2544     E.un = E.removeListener;
2545
2546     if (document && document.body) {
2547         E._load();
2548     } else {
2549         E.doAdd(window, "load", E._load);
2550     }
2551     E.doAdd(window, "unload", E._unload);
2552     E._tryPreloadAttach();
2553 })();
2554
2555 /*
2556  * Portions of this file are based on pieces of Yahoo User Interface Library
2557  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2558  * YUI licensed under the BSD License:
2559  * http://developer.yahoo.net/yui/license.txt
2560  * <script type="text/javascript">
2561  *
2562  */
2563
2564 (function() {
2565     /**
2566      * @class Roo.lib.Ajax
2567      *
2568      */
2569     Roo.lib.Ajax = {
2570         /**
2571          * @static 
2572          */
2573         request : function(method, uri, cb, data, options) {
2574             if(options){
2575                 var hs = options.headers;
2576                 if(hs){
2577                     for(var h in hs){
2578                         if(hs.hasOwnProperty(h)){
2579                             this.initHeader(h, hs[h], false);
2580                         }
2581                     }
2582                 }
2583                 if(options.xmlData){
2584                     this.initHeader('Content-Type', 'text/xml', false);
2585                     method = 'POST';
2586                     data = options.xmlData;
2587                 }
2588             }
2589
2590             return this.asyncRequest(method, uri, cb, data);
2591         },
2592
2593         serializeForm : function(form) {
2594             if(typeof form == 'string') {
2595                 form = (document.getElementById(form) || document.forms[form]);
2596             }
2597
2598             var el, name, val, disabled, data = '', hasSubmit = false;
2599             for (var i = 0; i < form.elements.length; i++) {
2600                 el = form.elements[i];
2601                 disabled = form.elements[i].disabled;
2602                 name = form.elements[i].name;
2603                 val = form.elements[i].value;
2604
2605                 if (!disabled && name){
2606                     switch (el.type)
2607                             {
2608                         case 'select-one':
2609                         case 'select-multiple':
2610                             for (var j = 0; j < el.options.length; j++) {
2611                                 if (el.options[j].selected) {
2612                                     if (Roo.isIE) {
2613                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].attributes['value'].specified ? el.options[j].value : el.options[j].text) + '&';
2614                                     }
2615                                     else {
2616                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].hasAttribute('value') ? el.options[j].value : el.options[j].text) + '&';
2617                                     }
2618                                 }
2619                             }
2620                             break;
2621                         case 'radio':
2622                         case 'checkbox':
2623                             if (el.checked) {
2624                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2625                             }
2626                             break;
2627                         case 'file':
2628
2629                         case undefined:
2630
2631                         case 'reset':
2632
2633                         case 'button':
2634
2635                             break;
2636                         case 'submit':
2637                             if(hasSubmit == false) {
2638                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2639                                 hasSubmit = true;
2640                             }
2641                             break;
2642                         default:
2643                             data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2644                             break;
2645                     }
2646                 }
2647             }
2648             data = data.substr(0, data.length - 1);
2649             return data;
2650         },
2651
2652         headers:{},
2653
2654         hasHeaders:false,
2655
2656         useDefaultHeader:true,
2657
2658         defaultPostHeader:'application/x-www-form-urlencoded',
2659
2660         useDefaultXhrHeader:true,
2661
2662         defaultXhrHeader:'XMLHttpRequest',
2663
2664         hasDefaultHeaders:true,
2665
2666         defaultHeaders:{},
2667
2668         poll:{},
2669
2670         timeout:{},
2671
2672         pollInterval:50,
2673
2674         transactionId:0,
2675
2676         setProgId:function(id)
2677         {
2678             this.activeX.unshift(id);
2679         },
2680
2681         setDefaultPostHeader:function(b)
2682         {
2683             this.useDefaultHeader = b;
2684         },
2685
2686         setDefaultXhrHeader:function(b)
2687         {
2688             this.useDefaultXhrHeader = b;
2689         },
2690
2691         setPollingInterval:function(i)
2692         {
2693             if (typeof i == 'number' && isFinite(i)) {
2694                 this.pollInterval = i;
2695             }
2696         },
2697
2698         createXhrObject:function(transactionId)
2699         {
2700             var obj,http;
2701             try
2702             {
2703
2704                 http = new XMLHttpRequest();
2705
2706                 obj = { conn:http, tId:transactionId };
2707             }
2708             catch(e)
2709             {
2710                 for (var i = 0; i < this.activeX.length; ++i) {
2711                     try
2712                     {
2713
2714                         http = new ActiveXObject(this.activeX[i]);
2715
2716                         obj = { conn:http, tId:transactionId };
2717                         break;
2718                     }
2719                     catch(e) {
2720                     }
2721                 }
2722             }
2723             finally
2724             {
2725                 return obj;
2726             }
2727         },
2728
2729         getConnectionObject:function()
2730         {
2731             var o;
2732             var tId = this.transactionId;
2733
2734             try
2735             {
2736                 o = this.createXhrObject(tId);
2737                 if (o) {
2738                     this.transactionId++;
2739                 }
2740             }
2741             catch(e) {
2742             }
2743             finally
2744             {
2745                 return o;
2746             }
2747         },
2748
2749         asyncRequest:function(method, uri, callback, postData)
2750         {
2751             var o = this.getConnectionObject();
2752
2753             if (!o) {
2754                 return null;
2755             }
2756             else {
2757                 o.conn.open(method, uri, true);
2758
2759                 if (this.useDefaultXhrHeader) {
2760                     if (!this.defaultHeaders['X-Requested-With']) {
2761                         this.initHeader('X-Requested-With', this.defaultXhrHeader, true);
2762                     }
2763                 }
2764
2765                 if(postData && this.useDefaultHeader){
2766                     this.initHeader('Content-Type', this.defaultPostHeader);
2767                 }
2768
2769                  if (this.hasDefaultHeaders || this.hasHeaders) {
2770                     this.setHeader(o);
2771                 }
2772
2773                 this.handleReadyState(o, callback);
2774                 o.conn.send(postData || null);
2775
2776                 return o;
2777             }
2778         },
2779
2780         handleReadyState:function(o, callback)
2781         {
2782             var oConn = this;
2783
2784             if (callback && callback.timeout) {
2785                 
2786                 this.timeout[o.tId] = window.setTimeout(function() {
2787                     oConn.abort(o, callback, true);
2788                 }, callback.timeout);
2789             }
2790
2791             this.poll[o.tId] = window.setInterval(
2792                     function() {
2793                         if (o.conn && o.conn.readyState == 4) {
2794                             window.clearInterval(oConn.poll[o.tId]);
2795                             delete oConn.poll[o.tId];
2796
2797                             if(callback && callback.timeout) {
2798                                 window.clearTimeout(oConn.timeout[o.tId]);
2799                                 delete oConn.timeout[o.tId];
2800                             }
2801
2802                             oConn.handleTransactionResponse(o, callback);
2803                         }
2804                     }
2805                     , this.pollInterval);
2806         },
2807
2808         handleTransactionResponse:function(o, callback, isAbort)
2809         {
2810
2811             if (!callback) {
2812                 this.releaseObject(o);
2813                 return;
2814             }
2815
2816             var httpStatus, responseObject;
2817
2818             try
2819             {
2820                 if (o.conn.status !== undefined && o.conn.status != 0) {
2821                     httpStatus = o.conn.status;
2822                 }
2823                 else {
2824                     httpStatus = 13030;
2825                 }
2826             }
2827             catch(e) {
2828
2829
2830                 httpStatus = 13030;
2831             }
2832
2833             if (httpStatus >= 200 && httpStatus < 300) {
2834                 responseObject = this.createResponseObject(o, callback.argument);
2835                 if (callback.success) {
2836                     if (!callback.scope) {
2837                         callback.success(responseObject);
2838                     }
2839                     else {
2840
2841
2842                         callback.success.apply(callback.scope, [responseObject]);
2843                     }
2844                 }
2845             }
2846             else {
2847                 switch (httpStatus) {
2848
2849                     case 12002:
2850                     case 12029:
2851                     case 12030:
2852                     case 12031:
2853                     case 12152:
2854                     case 13030:
2855                         responseObject = this.createExceptionObject(o.tId, callback.argument, (isAbort ? isAbort : false));
2856                         if (callback.failure) {
2857                             if (!callback.scope) {
2858                                 callback.failure(responseObject);
2859                             }
2860                             else {
2861                                 callback.failure.apply(callback.scope, [responseObject]);
2862                             }
2863                         }
2864                         break;
2865                     default:
2866                         responseObject = this.createResponseObject(o, callback.argument);
2867                         if (callback.failure) {
2868                             if (!callback.scope) {
2869                                 callback.failure(responseObject);
2870                             }
2871                             else {
2872                                 callback.failure.apply(callback.scope, [responseObject]);
2873                             }
2874                         }
2875                 }
2876             }
2877
2878             this.releaseObject(o);
2879             responseObject = null;
2880         },
2881
2882         createResponseObject:function(o, callbackArg)
2883         {
2884             var obj = {};
2885             var headerObj = {};
2886
2887             try
2888             {
2889                 var headerStr = o.conn.getAllResponseHeaders();
2890                 var header = headerStr.split('\n');
2891                 for (var i = 0; i < header.length; i++) {
2892                     var delimitPos = header[i].indexOf(':');
2893                     if (delimitPos != -1) {
2894                         headerObj[header[i].substring(0, delimitPos)] = header[i].substring(delimitPos + 2);
2895                     }
2896                 }
2897             }
2898             catch(e) {
2899             }
2900
2901             obj.tId = o.tId;
2902             obj.status = o.conn.status;
2903             obj.statusText = o.conn.statusText;
2904             obj.getResponseHeader = headerObj;
2905             obj.getAllResponseHeaders = headerStr;
2906             obj.responseText = o.conn.responseText;
2907             obj.responseXML = o.conn.responseXML;
2908
2909             if (typeof callbackArg !== undefined) {
2910                 obj.argument = callbackArg;
2911             }
2912
2913             return obj;
2914         },
2915
2916         createExceptionObject:function(tId, callbackArg, isAbort)
2917         {
2918             var COMM_CODE = 0;
2919             var COMM_ERROR = 'communication failure';
2920             var ABORT_CODE = -1;
2921             var ABORT_ERROR = 'transaction aborted';
2922
2923             var obj = {};
2924
2925             obj.tId = tId;
2926             if (isAbort) {
2927                 obj.status = ABORT_CODE;
2928                 obj.statusText = ABORT_ERROR;
2929             }
2930             else {
2931                 obj.status = COMM_CODE;
2932                 obj.statusText = COMM_ERROR;
2933             }
2934
2935             if (callbackArg) {
2936                 obj.argument = callbackArg;
2937             }
2938
2939             return obj;
2940         },
2941
2942         initHeader:function(label, value, isDefault)
2943         {
2944             var headerObj = (isDefault) ? this.defaultHeaders : this.headers;
2945
2946             if (headerObj[label] === undefined) {
2947                 headerObj[label] = value;
2948             }
2949             else {
2950
2951
2952                 headerObj[label] = value + "," + headerObj[label];
2953             }
2954
2955             if (isDefault) {
2956                 this.hasDefaultHeaders = true;
2957             }
2958             else {
2959                 this.hasHeaders = true;
2960             }
2961         },
2962
2963
2964         setHeader:function(o)
2965         {
2966             if (this.hasDefaultHeaders) {
2967                 for (var prop in this.defaultHeaders) {
2968                     if (this.defaultHeaders.hasOwnProperty(prop)) {
2969                         o.conn.setRequestHeader(prop, this.defaultHeaders[prop]);
2970                     }
2971                 }
2972             }
2973
2974             if (this.hasHeaders) {
2975                 for (var prop in this.headers) {
2976                     if (this.headers.hasOwnProperty(prop)) {
2977                         o.conn.setRequestHeader(prop, this.headers[prop]);
2978                     }
2979                 }
2980                 this.headers = {};
2981                 this.hasHeaders = false;
2982             }
2983         },
2984
2985         resetDefaultHeaders:function() {
2986             delete this.defaultHeaders;
2987             this.defaultHeaders = {};
2988             this.hasDefaultHeaders = false;
2989         },
2990
2991         abort:function(o, callback, isTimeout)
2992         {
2993             if(this.isCallInProgress(o)) {
2994                 o.conn.abort();
2995                 window.clearInterval(this.poll[o.tId]);
2996                 delete this.poll[o.tId];
2997                 if (isTimeout) {
2998                     delete this.timeout[o.tId];
2999                 }
3000
3001                 this.handleTransactionResponse(o, callback, true);
3002
3003                 return true;
3004             }
3005             else {
3006                 return false;
3007             }
3008         },
3009
3010
3011         isCallInProgress:function(o)
3012         {
3013             if (o && o.conn) {
3014                 return o.conn.readyState != 4 && o.conn.readyState != 0;
3015             }
3016             else {
3017
3018                 return false;
3019             }
3020         },
3021
3022
3023         releaseObject:function(o)
3024         {
3025
3026             o.conn = null;
3027
3028             o = null;
3029         },
3030
3031         activeX:[
3032         'MSXML2.XMLHTTP.3.0',
3033         'MSXML2.XMLHTTP',
3034         'Microsoft.XMLHTTP'
3035         ]
3036
3037
3038     };
3039 })();/*
3040  * Portions of this file are based on pieces of Yahoo User Interface Library
3041  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3042  * YUI licensed under the BSD License:
3043  * http://developer.yahoo.net/yui/license.txt
3044  * <script type="text/javascript">
3045  *
3046  */
3047
3048 Roo.lib.Region = function(t, r, b, l) {
3049     this.top = t;
3050     this[1] = t;
3051     this.right = r;
3052     this.bottom = b;
3053     this.left = l;
3054     this[0] = l;
3055 };
3056
3057
3058 Roo.lib.Region.prototype = {
3059     contains : function(region) {
3060         return ( region.left >= this.left &&
3061                  region.right <= this.right &&
3062                  region.top >= this.top &&
3063                  region.bottom <= this.bottom    );
3064
3065     },
3066
3067     getArea : function() {
3068         return ( (this.bottom - this.top) * (this.right - this.left) );
3069     },
3070
3071     intersect : function(region) {
3072         var t = Math.max(this.top, region.top);
3073         var r = Math.min(this.right, region.right);
3074         var b = Math.min(this.bottom, region.bottom);
3075         var l = Math.max(this.left, region.left);
3076
3077         if (b >= t && r >= l) {
3078             return new Roo.lib.Region(t, r, b, l);
3079         } else {
3080             return null;
3081         }
3082     },
3083     union : function(region) {
3084         var t = Math.min(this.top, region.top);
3085         var r = Math.max(this.right, region.right);
3086         var b = Math.max(this.bottom, region.bottom);
3087         var l = Math.min(this.left, region.left);
3088
3089         return new Roo.lib.Region(t, r, b, l);
3090     },
3091
3092     adjust : function(t, l, b, r) {
3093         this.top += t;
3094         this.left += l;
3095         this.right += r;
3096         this.bottom += b;
3097         return this;
3098     }
3099 };
3100
3101 Roo.lib.Region.getRegion = function(el) {
3102     var p = Roo.lib.Dom.getXY(el);
3103
3104     var t = p[1];
3105     var r = p[0] + el.offsetWidth;
3106     var b = p[1] + el.offsetHeight;
3107     var l = p[0];
3108
3109     return new Roo.lib.Region(t, r, b, l);
3110 };
3111 /*
3112  * Portions of this file are based on pieces of Yahoo User Interface Library
3113  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3114  * YUI licensed under the BSD License:
3115  * http://developer.yahoo.net/yui/license.txt
3116  * <script type="text/javascript">
3117  *
3118  */
3119 //@@dep Roo.lib.Region
3120
3121
3122 Roo.lib.Point = function(x, y) {
3123     if (x instanceof Array) {
3124         y = x[1];
3125         x = x[0];
3126     }
3127     this.x = this.right = this.left = this[0] = x;
3128     this.y = this.top = this.bottom = this[1] = y;
3129 };
3130
3131 Roo.lib.Point.prototype = new Roo.lib.Region();
3132 /*
3133  * Portions of this file are based on pieces of Yahoo User Interface Library
3134  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3135  * YUI licensed under the BSD License:
3136  * http://developer.yahoo.net/yui/license.txt
3137  * <script type="text/javascript">
3138  *
3139  */
3140  
3141 (function() {   
3142
3143     Roo.lib.Anim = {
3144         scroll : function(el, args, duration, easing, cb, scope) {
3145             this.run(el, args, duration, easing, cb, scope, Roo.lib.Scroll);
3146         },
3147
3148         motion : function(el, args, duration, easing, cb, scope) {
3149             this.run(el, args, duration, easing, cb, scope, Roo.lib.Motion);
3150         },
3151
3152         color : function(el, args, duration, easing, cb, scope) {
3153             this.run(el, args, duration, easing, cb, scope, Roo.lib.ColorAnim);
3154         },
3155
3156         run : function(el, args, duration, easing, cb, scope, type) {
3157             type = type || Roo.lib.AnimBase;
3158             if (typeof easing == "string") {
3159                 easing = Roo.lib.Easing[easing];
3160             }
3161             var anim = new type(el, args, duration, easing);
3162             anim.animateX(function() {
3163                 Roo.callback(cb, scope);
3164             });
3165             return anim;
3166         }
3167     };
3168 })();/*
3169  * Portions of this file are based on pieces of Yahoo User Interface Library
3170  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3171  * YUI licensed under the BSD License:
3172  * http://developer.yahoo.net/yui/license.txt
3173  * <script type="text/javascript">
3174  *
3175  */
3176
3177 (function() {    
3178     var libFlyweight;
3179     
3180     function fly(el) {
3181         if (!libFlyweight) {
3182             libFlyweight = new Roo.Element.Flyweight();
3183         }
3184         libFlyweight.dom = el;
3185         return libFlyweight;
3186     }
3187
3188     // since this uses fly! - it cant be in DOM (which does not have fly yet..)
3189     
3190    
3191     
3192     Roo.lib.AnimBase = function(el, attributes, duration, method) {
3193         if (el) {
3194             this.init(el, attributes, duration, method);
3195         }
3196     };
3197
3198     Roo.lib.AnimBase.fly = fly;
3199     
3200     
3201     
3202     Roo.lib.AnimBase.prototype = {
3203
3204         toString: function() {
3205             var el = this.getEl();
3206             var id = el.id || el.tagName;
3207             return ("Anim " + id);
3208         },
3209
3210         patterns: {
3211             noNegatives:        /width|height|opacity|padding/i,
3212             offsetAttribute:  /^((width|height)|(top|left))$/,
3213             defaultUnit:        /width|height|top$|bottom$|left$|right$/i,
3214             offsetUnit:         /\d+(em|%|en|ex|pt|in|cm|mm|pc)$/i
3215         },
3216
3217
3218         doMethod: function(attr, start, end) {
3219             return this.method(this.currentFrame, start, end - start, this.totalFrames);
3220         },
3221
3222
3223         setAttribute: function(attr, val, unit) {
3224             if (this.patterns.noNegatives.test(attr)) {
3225                 val = (val > 0) ? val : 0;
3226             }
3227
3228             Roo.fly(this.getEl(), '_anim').setStyle(attr, val + unit);
3229         },
3230
3231
3232         getAttribute: function(attr) {
3233             var el = this.getEl();
3234             var val = fly(el).getStyle(attr);
3235
3236             if (val !== 'auto' && !this.patterns.offsetUnit.test(val)) {
3237                 return parseFloat(val);
3238             }
3239
3240             var a = this.patterns.offsetAttribute.exec(attr) || [];
3241             var pos = !!( a[3] );
3242             var box = !!( a[2] );
3243
3244
3245             if (box || (fly(el).getStyle('position') == 'absolute' && pos)) {
3246                 val = el['offset' + a[0].charAt(0).toUpperCase() + a[0].substr(1)];
3247             } else {
3248                 val = 0;
3249             }
3250
3251             return val;
3252         },
3253
3254
3255         getDefaultUnit: function(attr) {
3256             if (this.patterns.defaultUnit.test(attr)) {
3257                 return 'px';
3258             }
3259
3260             return '';
3261         },
3262
3263         animateX : function(callback, scope) {
3264             var f = function() {
3265                 this.onComplete.removeListener(f);
3266                 if (typeof callback == "function") {
3267                     callback.call(scope || this, this);
3268                 }
3269             };
3270             this.onComplete.addListener(f, this);
3271             this.animate();
3272         },
3273
3274
3275         setRuntimeAttribute: function(attr) {
3276             var start;
3277             var end;
3278             var attributes = this.attributes;
3279
3280             this.runtimeAttributes[attr] = {};
3281
3282             var isset = function(prop) {
3283                 return (typeof prop !== 'undefined');
3284             };
3285
3286             if (!isset(attributes[attr]['to']) && !isset(attributes[attr]['by'])) {
3287                 return false;
3288             }
3289
3290             start = ( isset(attributes[attr]['from']) ) ? attributes[attr]['from'] : this.getAttribute(attr);
3291
3292
3293             if (isset(attributes[attr]['to'])) {
3294                 end = attributes[attr]['to'];
3295             } else if (isset(attributes[attr]['by'])) {
3296                 if (start.constructor == Array) {
3297                     end = [];
3298                     for (var i = 0, len = start.length; i < len; ++i) {
3299                         end[i] = start[i] + attributes[attr]['by'][i];
3300                     }
3301                 } else {
3302                     end = start + attributes[attr]['by'];
3303                 }
3304             }
3305
3306             this.runtimeAttributes[attr].start = start;
3307             this.runtimeAttributes[attr].end = end;
3308
3309
3310             this.runtimeAttributes[attr].unit = ( isset(attributes[attr].unit) ) ? attributes[attr]['unit'] : this.getDefaultUnit(attr);
3311         },
3312
3313
3314         init: function(el, attributes, duration, method) {
3315
3316             var isAnimated = false;
3317
3318
3319             var startTime = null;
3320
3321
3322             var actualFrames = 0;
3323
3324
3325             el = Roo.getDom(el);
3326
3327
3328             this.attributes = attributes || {};
3329
3330
3331             this.duration = duration || 1;
3332
3333
3334             this.method = method || Roo.lib.Easing.easeNone;
3335
3336
3337             this.useSeconds = true;
3338
3339
3340             this.currentFrame = 0;
3341
3342
3343             this.totalFrames = Roo.lib.AnimMgr.fps;
3344
3345
3346             this.getEl = function() {
3347                 return el;
3348             };
3349
3350
3351             this.isAnimated = function() {
3352                 return isAnimated;
3353             };
3354
3355
3356             this.getStartTime = function() {
3357                 return startTime;
3358             };
3359
3360             this.runtimeAttributes = {};
3361
3362
3363             this.animate = function() {
3364                 if (this.isAnimated()) {
3365                     return false;
3366                 }
3367
3368                 this.currentFrame = 0;
3369
3370                 this.totalFrames = ( this.useSeconds ) ? Math.ceil(Roo.lib.AnimMgr.fps * this.duration) : this.duration;
3371
3372                 Roo.lib.AnimMgr.registerElement(this);
3373             };
3374
3375
3376             this.stop = function(finish) {
3377                 if (finish) {
3378                     this.currentFrame = this.totalFrames;
3379                     this._onTween.fire();
3380                 }
3381                 Roo.lib.AnimMgr.stop(this);
3382             };
3383
3384             var onStart = function() {
3385                 this.onStart.fire();
3386
3387                 this.runtimeAttributes = {};
3388                 for (var attr in this.attributes) {
3389                     this.setRuntimeAttribute(attr);
3390                 }
3391
3392                 isAnimated = true;
3393                 actualFrames = 0;
3394                 startTime = new Date();
3395             };
3396
3397
3398             var onTween = function() {
3399                 var data = {
3400                     duration: new Date() - this.getStartTime(),
3401                     currentFrame: this.currentFrame
3402                 };
3403
3404                 data.toString = function() {
3405                     return (
3406                             'duration: ' + data.duration +
3407                             ', currentFrame: ' + data.currentFrame
3408                             );
3409                 };
3410
3411                 this.onTween.fire(data);
3412
3413                 var runtimeAttributes = this.runtimeAttributes;
3414
3415                 for (var attr in runtimeAttributes) {
3416                     this.setAttribute(attr, this.doMethod(attr, runtimeAttributes[attr].start, runtimeAttributes[attr].end), runtimeAttributes[attr].unit);
3417                 }
3418
3419                 actualFrames += 1;
3420             };
3421
3422             var onComplete = function() {
3423                 var actual_duration = (new Date() - startTime) / 1000 ;
3424
3425                 var data = {
3426                     duration: actual_duration,
3427                     frames: actualFrames,
3428                     fps: actualFrames / actual_duration
3429                 };
3430
3431                 data.toString = function() {
3432                     return (
3433                             'duration: ' + data.duration +
3434                             ', frames: ' + data.frames +
3435                             ', fps: ' + data.fps
3436                             );
3437                 };
3438
3439                 isAnimated = false;
3440                 actualFrames = 0;
3441                 this.onComplete.fire(data);
3442             };
3443
3444
3445             this._onStart = new Roo.util.Event(this);
3446             this.onStart = new Roo.util.Event(this);
3447             this.onTween = new Roo.util.Event(this);
3448             this._onTween = new Roo.util.Event(this);
3449             this.onComplete = new Roo.util.Event(this);
3450             this._onComplete = new Roo.util.Event(this);
3451             this._onStart.addListener(onStart);
3452             this._onTween.addListener(onTween);
3453             this._onComplete.addListener(onComplete);
3454         }
3455     };
3456 })();
3457 /*
3458  * Portions of this file are based on pieces of Yahoo User Interface Library
3459  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3460  * YUI licensed under the BSD License:
3461  * http://developer.yahoo.net/yui/license.txt
3462  * <script type="text/javascript">
3463  *
3464  */
3465
3466 Roo.lib.AnimMgr = new function() {
3467
3468     var thread = null;
3469
3470
3471     var queue = [];
3472
3473
3474     var tweenCount = 0;
3475
3476
3477     this.fps = 1000;
3478
3479
3480     this.delay = 1;
3481
3482
3483     this.registerElement = function(tween) {
3484         queue[queue.length] = tween;
3485         tweenCount += 1;
3486         tween._onStart.fire();
3487         this.start();
3488     };
3489
3490
3491     this.unRegister = function(tween, index) {
3492         tween._onComplete.fire();
3493         index = index || getIndex(tween);
3494         if (index != -1) {
3495             queue.splice(index, 1);
3496         }
3497
3498         tweenCount -= 1;
3499         if (tweenCount <= 0) {
3500             this.stop();
3501         }
3502     };
3503
3504
3505     this.start = function() {
3506         if (thread === null) {
3507             thread = setInterval(this.run, this.delay);
3508         }
3509     };
3510
3511
3512     this.stop = function(tween) {
3513         if (!tween) {
3514             clearInterval(thread);
3515
3516             for (var i = 0, len = queue.length; i < len; ++i) {
3517                 if (queue[0].isAnimated()) {
3518                     this.unRegister(queue[0], 0);
3519                 }
3520             }
3521
3522             queue = [];
3523             thread = null;
3524             tweenCount = 0;
3525         }
3526         else {
3527             this.unRegister(tween);
3528         }
3529     };
3530
3531
3532     this.run = function() {
3533         for (var i = 0, len = queue.length; i < len; ++i) {
3534             var tween = queue[i];
3535             if (!tween || !tween.isAnimated()) {
3536                 continue;
3537             }
3538
3539             if (tween.currentFrame < tween.totalFrames || tween.totalFrames === null)
3540             {
3541                 tween.currentFrame += 1;
3542
3543                 if (tween.useSeconds) {
3544                     correctFrame(tween);
3545                 }
3546                 tween._onTween.fire();
3547             }
3548             else {
3549                 Roo.lib.AnimMgr.stop(tween, i);
3550             }
3551         }
3552     };
3553
3554     var getIndex = function(anim) {
3555         for (var i = 0, len = queue.length; i < len; ++i) {
3556             if (queue[i] == anim) {
3557                 return i;
3558             }
3559         }
3560         return -1;
3561     };
3562
3563
3564     var correctFrame = function(tween) {
3565         var frames = tween.totalFrames;
3566         var frame = tween.currentFrame;
3567         var expected = (tween.currentFrame * tween.duration * 1000 / tween.totalFrames);
3568         var elapsed = (new Date() - tween.getStartTime());
3569         var tweak = 0;
3570
3571         if (elapsed < tween.duration * 1000) {
3572             tweak = Math.round((elapsed / expected - 1) * tween.currentFrame);
3573         } else {
3574             tweak = frames - (frame + 1);
3575         }
3576         if (tweak > 0 && isFinite(tweak)) {
3577             if (tween.currentFrame + tweak >= frames) {
3578                 tweak = frames - (frame + 1);
3579             }
3580
3581             tween.currentFrame += tweak;
3582         }
3583     };
3584 };
3585
3586     /*
3587  * Portions of this file are based on pieces of Yahoo User Interface Library
3588  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3589  * YUI licensed under the BSD License:
3590  * http://developer.yahoo.net/yui/license.txt
3591  * <script type="text/javascript">
3592  *
3593  */
3594 Roo.lib.Bezier = new function() {
3595
3596         this.getPosition = function(points, t) {
3597             var n = points.length;
3598             var tmp = [];
3599
3600             for (var i = 0; i < n; ++i) {
3601                 tmp[i] = [points[i][0], points[i][1]];
3602             }
3603
3604             for (var j = 1; j < n; ++j) {
3605                 for (i = 0; i < n - j; ++i) {
3606                     tmp[i][0] = (1 - t) * tmp[i][0] + t * tmp[parseInt(i + 1, 10)][0];
3607                     tmp[i][1] = (1 - t) * tmp[i][1] + t * tmp[parseInt(i + 1, 10)][1];
3608                 }
3609             }
3610
3611             return [ tmp[0][0], tmp[0][1] ];
3612
3613         };
3614     };/*
3615  * Portions of this file are based on pieces of Yahoo User Interface Library
3616  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3617  * YUI licensed under the BSD License:
3618  * http://developer.yahoo.net/yui/license.txt
3619  * <script type="text/javascript">
3620  *
3621  */
3622 (function() {
3623
3624     Roo.lib.ColorAnim = function(el, attributes, duration, method) {
3625         Roo.lib.ColorAnim.superclass.constructor.call(this, el, attributes, duration, method);
3626     };
3627
3628     Roo.extend(Roo.lib.ColorAnim, Roo.lib.AnimBase);
3629
3630     var fly = Roo.lib.AnimBase.fly;
3631     var Y = Roo.lib;
3632     var superclass = Y.ColorAnim.superclass;
3633     var proto = Y.ColorAnim.prototype;
3634
3635     proto.toString = function() {
3636         var el = this.getEl();
3637         var id = el.id || el.tagName;
3638         return ("ColorAnim " + id);
3639     };
3640
3641     proto.patterns.color = /color$/i;
3642     proto.patterns.rgb = /^rgb\(([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\)$/i;
3643     proto.patterns.hex = /^#?([0-9A-F]{2})([0-9A-F]{2})([0-9A-F]{2})$/i;
3644     proto.patterns.hex3 = /^#?([0-9A-F]{1})([0-9A-F]{1})([0-9A-F]{1})$/i;
3645     proto.patterns.transparent = /^transparent|rgba\(0, 0, 0, 0\)$/;
3646
3647
3648     proto.parseColor = function(s) {
3649         if (s.length == 3) {
3650             return s;
3651         }
3652
3653         var c = this.patterns.hex.exec(s);
3654         if (c && c.length == 4) {
3655             return [ parseInt(c[1], 16), parseInt(c[2], 16), parseInt(c[3], 16) ];
3656         }
3657
3658         c = this.patterns.rgb.exec(s);
3659         if (c && c.length == 4) {
3660             return [ parseInt(c[1], 10), parseInt(c[2], 10), parseInt(c[3], 10) ];
3661         }
3662
3663         c = this.patterns.hex3.exec(s);
3664         if (c && c.length == 4) {
3665             return [ parseInt(c[1] + c[1], 16), parseInt(c[2] + c[2], 16), parseInt(c[3] + c[3], 16) ];
3666         }
3667
3668         return null;
3669     };
3670     // since this uses fly! - it cant be in ColorAnim (which does not have fly yet..)
3671     proto.getAttribute = function(attr) {
3672         var el = this.getEl();
3673         if (this.patterns.color.test(attr)) {
3674             var val = fly(el).getStyle(attr);
3675
3676             if (this.patterns.transparent.test(val)) {
3677                 var parent = el.parentNode;
3678                 val = fly(parent).getStyle(attr);
3679
3680                 while (parent && this.patterns.transparent.test(val)) {
3681                     parent = parent.parentNode;
3682                     val = fly(parent).getStyle(attr);
3683                     if (parent.tagName.toUpperCase() == 'HTML') {
3684                         val = '#fff';
3685                     }
3686                 }
3687             }
3688         } else {
3689             val = superclass.getAttribute.call(this, attr);
3690         }
3691
3692         return val;
3693     };
3694     proto.getAttribute = function(attr) {
3695         var el = this.getEl();
3696         if (this.patterns.color.test(attr)) {
3697             var val = fly(el).getStyle(attr);
3698
3699             if (this.patterns.transparent.test(val)) {
3700                 var parent = el.parentNode;
3701                 val = fly(parent).getStyle(attr);
3702
3703                 while (parent && this.patterns.transparent.test(val)) {
3704                     parent = parent.parentNode;
3705                     val = fly(parent).getStyle(attr);
3706                     if (parent.tagName.toUpperCase() == 'HTML') {
3707                         val = '#fff';
3708                     }
3709                 }
3710             }
3711         } else {
3712             val = superclass.getAttribute.call(this, attr);
3713         }
3714
3715         return val;
3716     };
3717
3718     proto.doMethod = function(attr, start, end) {
3719         var val;
3720
3721         if (this.patterns.color.test(attr)) {
3722             val = [];
3723             for (var i = 0, len = start.length; i < len; ++i) {
3724                 val[i] = superclass.doMethod.call(this, attr, start[i], end[i]);
3725             }
3726
3727             val = 'rgb(' + Math.floor(val[0]) + ',' + Math.floor(val[1]) + ',' + Math.floor(val[2]) + ')';
3728         }
3729         else {
3730             val = superclass.doMethod.call(this, attr, start, end);
3731         }
3732
3733         return val;
3734     };
3735
3736     proto.setRuntimeAttribute = function(attr) {
3737         superclass.setRuntimeAttribute.call(this, attr);
3738
3739         if (this.patterns.color.test(attr)) {
3740             var attributes = this.attributes;
3741             var start = this.parseColor(this.runtimeAttributes[attr].start);
3742             var end = this.parseColor(this.runtimeAttributes[attr].end);
3743
3744             if (typeof attributes[attr]['to'] === 'undefined' && typeof attributes[attr]['by'] !== 'undefined') {
3745                 end = this.parseColor(attributes[attr].by);
3746
3747                 for (var i = 0, len = start.length; i < len; ++i) {
3748                     end[i] = start[i] + end[i];
3749                 }
3750             }
3751
3752             this.runtimeAttributes[attr].start = start;
3753             this.runtimeAttributes[attr].end = end;
3754         }
3755     };
3756 })();
3757
3758 /*
3759  * Portions of this file are based on pieces of Yahoo User Interface Library
3760  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3761  * YUI licensed under the BSD License:
3762  * http://developer.yahoo.net/yui/license.txt
3763  * <script type="text/javascript">
3764  *
3765  */
3766 Roo.lib.Easing = {
3767
3768
3769     easeNone: function (t, b, c, d) {
3770         return c * t / d + b;
3771     },
3772
3773
3774     easeIn: function (t, b, c, d) {
3775         return c * (t /= d) * t + b;
3776     },
3777
3778
3779     easeOut: function (t, b, c, d) {
3780         return -c * (t /= d) * (t - 2) + b;
3781     },
3782
3783
3784     easeBoth: function (t, b, c, d) {
3785         if ((t /= d / 2) < 1) {
3786             return c / 2 * t * t + b;
3787         }
3788
3789         return -c / 2 * ((--t) * (t - 2) - 1) + b;
3790     },
3791
3792
3793     easeInStrong: function (t, b, c, d) {
3794         return c * (t /= d) * t * t * t + b;
3795     },
3796
3797
3798     easeOutStrong: function (t, b, c, d) {
3799         return -c * ((t = t / d - 1) * t * t * t - 1) + b;
3800     },
3801
3802
3803     easeBothStrong: function (t, b, c, d) {
3804         if ((t /= d / 2) < 1) {
3805             return c / 2 * t * t * t * t + b;
3806         }
3807
3808         return -c / 2 * ((t -= 2) * t * t * t - 2) + b;
3809     },
3810
3811
3812
3813     elasticIn: function (t, b, c, d, a, p) {
3814         if (t == 0) {
3815             return b;
3816         }
3817         if ((t /= d) == 1) {
3818             return b + c;
3819         }
3820         if (!p) {
3821             p = d * .3;
3822         }
3823
3824         if (!a || a < Math.abs(c)) {
3825             a = c;
3826             var s = p / 4;
3827         }
3828         else {
3829             var s = p / (2 * Math.PI) * Math.asin(c / a);
3830         }
3831
3832         return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3833     },
3834
3835
3836     elasticOut: function (t, b, c, d, a, p) {
3837         if (t == 0) {
3838             return b;
3839         }
3840         if ((t /= d) == 1) {
3841             return b + c;
3842         }
3843         if (!p) {
3844             p = d * .3;
3845         }
3846
3847         if (!a || a < Math.abs(c)) {
3848             a = c;
3849             var s = p / 4;
3850         }
3851         else {
3852             var s = p / (2 * Math.PI) * Math.asin(c / a);
3853         }
3854
3855         return a * Math.pow(2, -10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b;
3856     },
3857
3858
3859     elasticBoth: function (t, b, c, d, a, p) {
3860         if (t == 0) {
3861             return b;
3862         }
3863
3864         if ((t /= d / 2) == 2) {
3865             return b + c;
3866         }
3867
3868         if (!p) {
3869             p = d * (.3 * 1.5);
3870         }
3871
3872         if (!a || a < Math.abs(c)) {
3873             a = c;
3874             var s = p / 4;
3875         }
3876         else {
3877             var s = p / (2 * Math.PI) * Math.asin(c / a);
3878         }
3879
3880         if (t < 1) {
3881             return -.5 * (a * Math.pow(2, 10 * (t -= 1)) *
3882                           Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3883         }
3884         return a * Math.pow(2, -10 * (t -= 1)) *
3885                Math.sin((t * d - s) * (2 * Math.PI) / p) * .5 + c + b;
3886     },
3887
3888
3889
3890     backIn: function (t, b, c, d, s) {
3891         if (typeof s == 'undefined') {
3892             s = 1.70158;
3893         }
3894         return c * (t /= d) * t * ((s + 1) * t - s) + b;
3895     },
3896
3897
3898     backOut: function (t, b, c, d, s) {
3899         if (typeof s == 'undefined') {
3900             s = 1.70158;
3901         }
3902         return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
3903     },
3904
3905
3906     backBoth: function (t, b, c, d, s) {
3907         if (typeof s == 'undefined') {
3908             s = 1.70158;
3909         }
3910
3911         if ((t /= d / 2 ) < 1) {
3912             return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
3913         }
3914         return c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
3915     },
3916
3917
3918     bounceIn: function (t, b, c, d) {
3919         return c - Roo.lib.Easing.bounceOut(d - t, 0, c, d) + b;
3920     },
3921
3922
3923     bounceOut: function (t, b, c, d) {
3924         if ((t /= d) < (1 / 2.75)) {
3925             return c * (7.5625 * t * t) + b;
3926         } else if (t < (2 / 2.75)) {
3927             return c * (7.5625 * (t -= (1.5 / 2.75)) * t + .75) + b;
3928         } else if (t < (2.5 / 2.75)) {
3929             return c * (7.5625 * (t -= (2.25 / 2.75)) * t + .9375) + b;
3930         }
3931         return c * (7.5625 * (t -= (2.625 / 2.75)) * t + .984375) + b;
3932     },
3933
3934
3935     bounceBoth: function (t, b, c, d) {
3936         if (t < d / 2) {
3937             return Roo.lib.Easing.bounceIn(t * 2, 0, c, d) * .5 + b;
3938         }
3939         return Roo.lib.Easing.bounceOut(t * 2 - d, 0, c, d) * .5 + c * .5 + b;
3940     }
3941 };/*
3942  * Portions of this file are based on pieces of Yahoo User Interface Library
3943  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3944  * YUI licensed under the BSD License:
3945  * http://developer.yahoo.net/yui/license.txt
3946  * <script type="text/javascript">
3947  *
3948  */
3949     (function() {
3950         Roo.lib.Motion = function(el, attributes, duration, method) {
3951             if (el) {
3952                 Roo.lib.Motion.superclass.constructor.call(this, el, attributes, duration, method);
3953             }
3954         };
3955
3956         Roo.extend(Roo.lib.Motion, Roo.lib.ColorAnim);
3957
3958
3959         var Y = Roo.lib;
3960         var superclass = Y.Motion.superclass;
3961         var proto = Y.Motion.prototype;
3962
3963         proto.toString = function() {
3964             var el = this.getEl();
3965             var id = el.id || el.tagName;
3966             return ("Motion " + id);
3967         };
3968
3969         proto.patterns.points = /^points$/i;
3970
3971         proto.setAttribute = function(attr, val, unit) {
3972             if (this.patterns.points.test(attr)) {
3973                 unit = unit || 'px';
3974                 superclass.setAttribute.call(this, 'left', val[0], unit);
3975                 superclass.setAttribute.call(this, 'top', val[1], unit);
3976             } else {
3977                 superclass.setAttribute.call(this, attr, val, unit);
3978             }
3979         };
3980
3981         proto.getAttribute = function(attr) {
3982             if (this.patterns.points.test(attr)) {
3983                 var val = [
3984                         superclass.getAttribute.call(this, 'left'),
3985                         superclass.getAttribute.call(this, 'top')
3986                         ];
3987             } else {
3988                 val = superclass.getAttribute.call(this, attr);
3989             }
3990
3991             return val;
3992         };
3993
3994         proto.doMethod = function(attr, start, end) {
3995             var val = null;
3996
3997             if (this.patterns.points.test(attr)) {
3998                 var t = this.method(this.currentFrame, 0, 100, this.totalFrames) / 100;
3999                 val = Y.Bezier.getPosition(this.runtimeAttributes[attr], t);
4000             } else {
4001                 val = superclass.doMethod.call(this, attr, start, end);
4002             }
4003             return val;
4004         };
4005
4006         proto.setRuntimeAttribute = function(attr) {
4007             if (this.patterns.points.test(attr)) {
4008                 var el = this.getEl();
4009                 var attributes = this.attributes;
4010                 var start;
4011                 var control = attributes['points']['control'] || [];
4012                 var end;
4013                 var i, len;
4014
4015                 if (control.length > 0 && !(control[0] instanceof Array)) {
4016                     control = [control];
4017                 } else {
4018                     var tmp = [];
4019                     for (i = 0,len = control.length; i < len; ++i) {
4020                         tmp[i] = control[i];
4021                     }
4022                     control = tmp;
4023                 }
4024
4025                 Roo.fly(el).position();
4026
4027                 if (isset(attributes['points']['from'])) {
4028                     Roo.lib.Dom.setXY(el, attributes['points']['from']);
4029                 }
4030                 else {
4031                     Roo.lib.Dom.setXY(el, Roo.lib.Dom.getXY(el));
4032                 }
4033
4034                 start = this.getAttribute('points');
4035
4036
4037                 if (isset(attributes['points']['to'])) {
4038                     end = translateValues.call(this, attributes['points']['to'], start);
4039
4040                     var pageXY = Roo.lib.Dom.getXY(this.getEl());
4041                     for (i = 0,len = control.length; i < len; ++i) {
4042                         control[i] = translateValues.call(this, control[i], start);
4043                     }
4044
4045
4046                 } else if (isset(attributes['points']['by'])) {
4047                     end = [ start[0] + attributes['points']['by'][0], start[1] + attributes['points']['by'][1] ];
4048
4049                     for (i = 0,len = control.length; i < len; ++i) {
4050                         control[i] = [ start[0] + control[i][0], start[1] + control[i][1] ];
4051                     }
4052                 }
4053
4054                 this.runtimeAttributes[attr] = [start];
4055
4056                 if (control.length > 0) {
4057                     this.runtimeAttributes[attr] = this.runtimeAttributes[attr].concat(control);
4058                 }
4059
4060                 this.runtimeAttributes[attr][this.runtimeAttributes[attr].length] = end;
4061             }
4062             else {
4063                 superclass.setRuntimeAttribute.call(this, attr);
4064             }
4065         };
4066
4067         var translateValues = function(val, start) {
4068             var pageXY = Roo.lib.Dom.getXY(this.getEl());
4069             val = [ val[0] - pageXY[0] + start[0], val[1] - pageXY[1] + start[1] ];
4070
4071             return val;
4072         };
4073
4074         var isset = function(prop) {
4075             return (typeof prop !== 'undefined');
4076         };
4077     })();
4078 /*
4079  * Portions of this file are based on pieces of Yahoo User Interface Library
4080  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
4081  * YUI licensed under the BSD License:
4082  * http://developer.yahoo.net/yui/license.txt
4083  * <script type="text/javascript">
4084  *
4085  */
4086     (function() {
4087         Roo.lib.Scroll = function(el, attributes, duration, method) {
4088             if (el) {
4089                 Roo.lib.Scroll.superclass.constructor.call(this, el, attributes, duration, method);
4090             }
4091         };
4092
4093         Roo.extend(Roo.lib.Scroll, Roo.lib.ColorAnim);
4094
4095
4096         var Y = Roo.lib;
4097         var superclass = Y.Scroll.superclass;
4098         var proto = Y.Scroll.prototype;
4099
4100         proto.toString = function() {
4101             var el = this.getEl();
4102             var id = el.id || el.tagName;
4103             return ("Scroll " + id);
4104         };
4105
4106         proto.doMethod = function(attr, start, end) {
4107             var val = null;
4108
4109             if (attr == 'scroll') {
4110                 val = [
4111                         this.method(this.currentFrame, start[0], end[0] - start[0], this.totalFrames),
4112                         this.method(this.currentFrame, start[1], end[1] - start[1], this.totalFrames)
4113                         ];
4114
4115             } else {
4116                 val = superclass.doMethod.call(this, attr, start, end);
4117             }
4118             return val;
4119         };
4120
4121         proto.getAttribute = function(attr) {
4122             var val = null;
4123             var el = this.getEl();
4124
4125             if (attr == 'scroll') {
4126                 val = [ el.scrollLeft, el.scrollTop ];
4127             } else {
4128                 val = superclass.getAttribute.call(this, attr);
4129             }
4130
4131             return val;
4132         };
4133
4134         proto.setAttribute = function(attr, val, unit) {
4135             var el = this.getEl();
4136
4137             if (attr == 'scroll') {
4138                 el.scrollLeft = val[0];
4139                 el.scrollTop = val[1];
4140             } else {
4141                 superclass.setAttribute.call(this, attr, val, unit);
4142             }
4143         };
4144     })();
4145 /*
4146  * Based on:
4147  * Ext JS Library 1.1.1
4148  * Copyright(c) 2006-2007, Ext JS, LLC.
4149  *
4150  * Originally Released Under LGPL - original licence link has changed is not relivant.
4151  *
4152  * Fork - LGPL
4153  * <script type="text/javascript">
4154  */
4155
4156
4157 // nasty IE9 hack - what a pile of crap that is..
4158
4159  if (typeof Range != "undefined" && typeof Range.prototype.createContextualFragment == "undefined") {
4160     Range.prototype.createContextualFragment = function (html) {
4161         var doc = window.document;
4162         var container = doc.createElement("div");
4163         container.innerHTML = html;
4164         var frag = doc.createDocumentFragment(), n;
4165         while ((n = container.firstChild)) {
4166             frag.appendChild(n);
4167         }
4168         return frag;
4169     };
4170 }
4171
4172 /**
4173  * @class Roo.DomHelper
4174  * Utility class for working with DOM and/or Templates. It transparently supports using HTML fragments or DOM.
4175  * For more information see <a href="http://web.archive.org/web/20071221063734/http://www.jackslocum.com/blog/2006/10/06/domhelper-create-elements-using-dom-html-fragments-or-templates/">this blog post with examples</a>.
4176  * @singleton
4177  */
4178 Roo.DomHelper = function(){
4179     var tempTableEl = null;
4180     var emptyTags = /^(?:br|frame|hr|img|input|link|meta|range|spacer|wbr|area|param|col)$/i;
4181     var tableRe = /^table|tbody|tr|td$/i;
4182     var xmlns = {};
4183     // build as innerHTML where available
4184     /** @ignore */
4185     var createHtml = function(o){
4186         if(typeof o == 'string'){
4187             return o;
4188         }
4189         var b = "";
4190         if(!o.tag){
4191             o.tag = "div";
4192         }
4193         b += "<" + o.tag;
4194         for(var attr in o){
4195             if(attr == "tag" || attr == "children" || attr == "cn" || attr == "html" || typeof o[attr] == "function") { continue; }
4196             if(attr == "style"){
4197                 var s = o["style"];
4198                 if(typeof s == "function"){
4199                     s = s.call();
4200                 }
4201                 if(typeof s == "string"){
4202                     b += ' style="' + s + '"';
4203                 }else if(typeof s == "object"){
4204                     b += ' style="';
4205                     for(var key in s){
4206                         if(typeof s[key] != "function"){
4207                             b += key + ":" + s[key] + ";";
4208                         }
4209                     }
4210                     b += '"';
4211                 }
4212             }else{
4213                 if(attr == "cls"){
4214                     b += ' class="' + o["cls"] + '"';
4215                 }else if(attr == "htmlFor"){
4216                     b += ' for="' + o["htmlFor"] + '"';
4217                 }else{
4218                     b += " " + attr + '="' + o[attr] + '"';
4219                 }
4220             }
4221         }
4222         if(emptyTags.test(o.tag)){
4223             b += "/>";
4224         }else{
4225             b += ">";
4226             var cn = o.children || o.cn;
4227             if(cn){
4228                 //http://bugs.kde.org/show_bug.cgi?id=71506
4229                 if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4230                     for(var i = 0, len = cn.length; i < len; i++) {
4231                         b += createHtml(cn[i], b);
4232                     }
4233                 }else{
4234                     b += createHtml(cn, b);
4235                 }
4236             }
4237             if(o.html){
4238                 b += o.html;
4239             }
4240             b += "</" + o.tag + ">";
4241         }
4242         return b;
4243     };
4244
4245     // build as dom
4246     /** @ignore */
4247     var createDom = function(o, parentNode){
4248          
4249         // defininition craeted..
4250         var ns = false;
4251         if (o.ns && o.ns != 'html') {
4252                
4253             if (o.xmlns && typeof(xmlns[o.ns]) == 'undefined') {
4254                 xmlns[o.ns] = o.xmlns;
4255                 ns = o.xmlns;
4256             }
4257             if (typeof(xmlns[o.ns]) == 'undefined') {
4258                 console.log("Trying to create namespace element " + o.ns + ", however no xmlns was sent to builder previously");
4259             }
4260             ns = xmlns[o.ns];
4261         }
4262         
4263         
4264         if (typeof(o) == 'string') {
4265             return parentNode.appendChild(document.createTextNode(o));
4266         }
4267         o.tag = o.tag || div;
4268         if (o.ns && Roo.isIE) {
4269             ns = false;
4270             o.tag = o.ns + ':' + o.tag;
4271             
4272         }
4273         var el = ns ? document.createElementNS( ns, o.tag||'div') :  document.createElement(o.tag||'div');
4274         var useSet = el.setAttribute ? true : false; // In IE some elements don't have setAttribute
4275         for(var attr in o){
4276             
4277             if(attr == "tag" || attr == "ns" ||attr == "xmlns" ||attr == "children" || attr == "cn" || attr == "html" || 
4278                     attr == "style" || typeof o[attr] == "function") { continue; }
4279                     
4280             if(attr=="cls" && Roo.isIE){
4281                 el.className = o["cls"];
4282             }else{
4283                 if(useSet) { el.setAttribute(attr=="cls" ? 'class' : attr, o[attr]);}
4284                 else { 
4285                     el[attr] = o[attr];
4286                 }
4287             }
4288         }
4289         Roo.DomHelper.applyStyles(el, o.style);
4290         var cn = o.children || o.cn;
4291         if(cn){
4292             //http://bugs.kde.org/show_bug.cgi?id=71506
4293              if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4294                 for(var i = 0, len = cn.length; i < len; i++) {
4295                     createDom(cn[i], el);
4296                 }
4297             }else{
4298                 createDom(cn, el);
4299             }
4300         }
4301         if(o.html){
4302             el.innerHTML = o.html;
4303         }
4304         if(parentNode){
4305            parentNode.appendChild(el);
4306         }
4307         return el;
4308     };
4309
4310     var ieTable = function(depth, s, h, e){
4311         tempTableEl.innerHTML = [s, h, e].join('');
4312         var i = -1, el = tempTableEl;
4313         while(++i < depth){
4314             el = el.firstChild;
4315         }
4316         return el;
4317     };
4318
4319     // kill repeat to save bytes
4320     var ts = '<table>',
4321         te = '</table>',
4322         tbs = ts+'<tbody>',
4323         tbe = '</tbody>'+te,
4324         trs = tbs + '<tr>',
4325         tre = '</tr>'+tbe;
4326
4327     /**
4328      * @ignore
4329      * Nasty code for IE's broken table implementation
4330      */
4331     var insertIntoTable = function(tag, where, el, html){
4332         if(!tempTableEl){
4333             tempTableEl = document.createElement('div');
4334         }
4335         var node;
4336         var before = null;
4337         if(tag == 'td'){
4338             if(where == 'afterbegin' || where == 'beforeend'){ // INTO a TD
4339                 return;
4340             }
4341             if(where == 'beforebegin'){
4342                 before = el;
4343                 el = el.parentNode;
4344             } else{
4345                 before = el.nextSibling;
4346                 el = el.parentNode;
4347             }
4348             node = ieTable(4, trs, html, tre);
4349         }
4350         else if(tag == 'tr'){
4351             if(where == 'beforebegin'){
4352                 before = el;
4353                 el = el.parentNode;
4354                 node = ieTable(3, tbs, html, tbe);
4355             } else if(where == 'afterend'){
4356                 before = el.nextSibling;
4357                 el = el.parentNode;
4358                 node = ieTable(3, tbs, html, tbe);
4359             } else{ // INTO a TR
4360                 if(where == 'afterbegin'){
4361                     before = el.firstChild;
4362                 }
4363                 node = ieTable(4, trs, html, tre);
4364             }
4365         } else if(tag == 'tbody'){
4366             if(where == 'beforebegin'){
4367                 before = el;
4368                 el = el.parentNode;
4369                 node = ieTable(2, ts, html, te);
4370             } else if(where == 'afterend'){
4371                 before = el.nextSibling;
4372                 el = el.parentNode;
4373                 node = ieTable(2, ts, html, te);
4374             } else{
4375                 if(where == 'afterbegin'){
4376                     before = el.firstChild;
4377                 }
4378                 node = ieTable(3, tbs, html, tbe);
4379             }
4380         } else{ // TABLE
4381             if(where == 'beforebegin' || where == 'afterend'){ // OUTSIDE the table
4382                 return;
4383             }
4384             if(where == 'afterbegin'){
4385                 before = el.firstChild;
4386             }
4387             node = ieTable(2, ts, html, te);
4388         }
4389         el.insertBefore(node, before);
4390         return node;
4391     };
4392
4393     return {
4394     /** True to force the use of DOM instead of html fragments @type Boolean */
4395     useDom : false,
4396
4397     /**
4398      * Returns the markup for the passed Element(s) config
4399      * @param {Object} o The Dom object spec (and children)
4400      * @return {String}
4401      */
4402     markup : function(o){
4403         return createHtml(o);
4404     },
4405
4406     /**
4407      * Applies a style specification to an element
4408      * @param {String/HTMLElement} el The element to apply styles to
4409      * @param {String/Object/Function} styles A style specification string eg "width:100px", or object in the form {width:"100px"}, or
4410      * a function which returns such a specification.
4411      */
4412     applyStyles : function(el, styles){
4413         if(styles){
4414            el = Roo.fly(el);
4415            if(typeof styles == "string"){
4416                var re = /\s?([a-z\-]*)\:\s?([^;]*);?/gi;
4417                var matches;
4418                while ((matches = re.exec(styles)) != null){
4419                    el.setStyle(matches[1], matches[2]);
4420                }
4421            }else if (typeof styles == "object"){
4422                for (var style in styles){
4423                   el.setStyle(style, styles[style]);
4424                }
4425            }else if (typeof styles == "function"){
4426                 Roo.DomHelper.applyStyles(el, styles.call());
4427            }
4428         }
4429     },
4430
4431     /**
4432      * Inserts an HTML fragment into the Dom
4433      * @param {String} where Where to insert the html in relation to el - beforeBegin, afterBegin, beforeEnd, afterEnd.
4434      * @param {HTMLElement} el The context element
4435      * @param {String} html The HTML fragmenet
4436      * @return {HTMLElement} The new node
4437      */
4438     insertHtml : function(where, el, html){
4439         where = where.toLowerCase();
4440         if(el.insertAdjacentHTML){
4441             if(tableRe.test(el.tagName)){
4442                 var rs;
4443                 if(rs = insertIntoTable(el.tagName.toLowerCase(), where, el, html)){
4444                     return rs;
4445                 }
4446             }
4447             switch(where){
4448                 case "beforebegin":
4449                     el.insertAdjacentHTML('BeforeBegin', html);
4450                     return el.previousSibling;
4451                 case "afterbegin":
4452                     el.insertAdjacentHTML('AfterBegin', html);
4453                     return el.firstChild;
4454                 case "beforeend":
4455                     el.insertAdjacentHTML('BeforeEnd', html);
4456                     return el.lastChild;
4457                 case "afterend":
4458                     el.insertAdjacentHTML('AfterEnd', html);
4459                     return el.nextSibling;
4460             }
4461             throw 'Illegal insertion point -> "' + where + '"';
4462         }
4463         var range = el.ownerDocument.createRange();
4464         var frag;
4465         switch(where){
4466              case "beforebegin":
4467                 range.setStartBefore(el);
4468                 frag = range.createContextualFragment(html);
4469                 el.parentNode.insertBefore(frag, el);
4470                 return el.previousSibling;
4471              case "afterbegin":
4472                 if(el.firstChild){
4473                     range.setStartBefore(el.firstChild);
4474                     frag = range.createContextualFragment(html);
4475                     el.insertBefore(frag, el.firstChild);
4476                     return el.firstChild;
4477                 }else{
4478                     el.innerHTML = html;
4479                     return el.firstChild;
4480                 }
4481             case "beforeend":
4482                 if(el.lastChild){
4483                     range.setStartAfter(el.lastChild);
4484                     frag = range.createContextualFragment(html);
4485                     el.appendChild(frag);
4486                     return el.lastChild;
4487                 }else{
4488                     el.innerHTML = html;
4489                     return el.lastChild;
4490                 }
4491             case "afterend":
4492                 range.setStartAfter(el);
4493                 frag = range.createContextualFragment(html);
4494                 el.parentNode.insertBefore(frag, el.nextSibling);
4495                 return el.nextSibling;
4496             }
4497             throw 'Illegal insertion point -> "' + where + '"';
4498     },
4499
4500     /**
4501      * Creates new Dom element(s) and inserts them before el
4502      * @param {String/HTMLElement/Element} el The context element
4503      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4504      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4505      * @return {HTMLElement/Roo.Element} The new node
4506      */
4507     insertBefore : function(el, o, returnElement){
4508         return this.doInsert(el, o, returnElement, "beforeBegin");
4509     },
4510
4511     /**
4512      * Creates new Dom element(s) and inserts them after el
4513      * @param {String/HTMLElement/Element} el The context element
4514      * @param {Object} o The Dom object spec (and children)
4515      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4516      * @return {HTMLElement/Roo.Element} The new node
4517      */
4518     insertAfter : function(el, o, returnElement){
4519         return this.doInsert(el, o, returnElement, "afterEnd", "nextSibling");
4520     },
4521
4522     /**
4523      * Creates new Dom element(s) and inserts them as the first child of el
4524      * @param {String/HTMLElement/Element} el The context element
4525      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4526      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4527      * @return {HTMLElement/Roo.Element} The new node
4528      */
4529     insertFirst : function(el, o, returnElement){
4530         return this.doInsert(el, o, returnElement, "afterBegin");
4531     },
4532
4533     // private
4534     doInsert : function(el, o, returnElement, pos, sibling){
4535         el = Roo.getDom(el);
4536         var newNode;
4537         if(this.useDom || o.ns){
4538             newNode = createDom(o, null);
4539             el.parentNode.insertBefore(newNode, sibling ? el[sibling] : el);
4540         }else{
4541             var html = createHtml(o);
4542             newNode = this.insertHtml(pos, el, html);
4543         }
4544         return returnElement ? Roo.get(newNode, true) : newNode;
4545     },
4546
4547     /**
4548      * Creates new Dom element(s) and appends them to el
4549      * @param {String/HTMLElement/Element} el The context element
4550      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4551      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4552      * @return {HTMLElement/Roo.Element} The new node
4553      */
4554     append : function(el, o, returnElement){
4555         el = Roo.getDom(el);
4556         var newNode;
4557         if(this.useDom || o.ns){
4558             newNode = createDom(o, null);
4559             el.appendChild(newNode);
4560         }else{
4561             var html = createHtml(o);
4562             newNode = this.insertHtml("beforeEnd", el, html);
4563         }
4564         return returnElement ? Roo.get(newNode, true) : newNode;
4565     },
4566
4567     /**
4568      * Creates new Dom element(s) and overwrites the contents of el with them
4569      * @param {String/HTMLElement/Element} el The context element
4570      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4571      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4572      * @return {HTMLElement/Roo.Element} The new node
4573      */
4574     overwrite : function(el, o, returnElement){
4575         el = Roo.getDom(el);
4576         if (o.ns) {
4577           
4578             while (el.childNodes.length) {
4579                 el.removeChild(el.firstChild);
4580             }
4581             createDom(o, el);
4582         } else {
4583             el.innerHTML = createHtml(o);   
4584         }
4585         
4586         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4587     },
4588
4589     /**
4590      * Creates a new Roo.DomHelper.Template from the Dom object spec
4591      * @param {Object} o The Dom object spec (and children)
4592      * @return {Roo.DomHelper.Template} The new template
4593      */
4594     createTemplate : function(o){
4595         var html = createHtml(o);
4596         return new Roo.Template(html);
4597     }
4598     };
4599 }();
4600 /*
4601  * Based on:
4602  * Ext JS Library 1.1.1
4603  * Copyright(c) 2006-2007, Ext JS, LLC.
4604  *
4605  * Originally Released Under LGPL - original licence link has changed is not relivant.
4606  *
4607  * Fork - LGPL
4608  * <script type="text/javascript">
4609  */
4610  
4611 /**
4612 * @class Roo.Template
4613 * Represents an HTML fragment template. Templates can be precompiled for greater performance.
4614 * For a list of available format functions, see {@link Roo.util.Format}.<br />
4615 * Usage:
4616 <pre><code>
4617 var t = new Roo.Template({
4618     html :  '&lt;div name="{id}"&gt;' + 
4619         '&lt;span class="{cls}"&gt;{name:trim} {someval:this.myformat}{value:ellipsis(10)}&lt;/span&gt;' +
4620         '&lt;/div&gt;',
4621     myformat: function (value, allValues) {
4622         return 'XX' + value;
4623     }
4624 });
4625 t.append('some-element', {id: 'myid', cls: 'myclass', name: 'foo', value: 'bar'});
4626 </code></pre>
4627 * For more information see this blog post with examples:
4628 *  <a href="http://www.cnitblog.com/seeyeah/archive/2011/12/30/38728.html/">DomHelper
4629      - Create Elements using DOM, HTML fragments and Templates</a>. 
4630 * @constructor
4631 * @param {Object} cfg - Configuration object.
4632 */
4633 Roo.Template = function(cfg){
4634     // BC!
4635     if(cfg instanceof Array){
4636         cfg = cfg.join("");
4637     }else if(arguments.length > 1){
4638         cfg = Array.prototype.join.call(arguments, "");
4639     }
4640     
4641     
4642     if (typeof(cfg) == 'object') {
4643         Roo.apply(this,cfg)
4644     } else {
4645         // bc
4646         this.html = cfg;
4647     }
4648     if (this.url) {
4649         this.load();
4650     }
4651     
4652 };
4653 Roo.Template.prototype = {
4654     
4655     /**
4656      * @cfg {String} url  The Url to load the template from. beware if you are loading from a url, the data may not be ready if you use it instantly..
4657      *                    it should be fixed so that template is observable...
4658      */
4659     url : false,
4660     /**
4661      * @cfg {String} html  The HTML fragment or an array of fragments to join("") or multiple arguments to join("")
4662      */
4663     html : '',
4664     /**
4665      * Returns an HTML fragment of this template with the specified values applied.
4666      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4667      * @return {String} The HTML fragment
4668      */
4669     applyTemplate : function(values){
4670         Roo.log(["applyTemplate", values]);
4671         try {
4672            
4673             if(this.compiled){
4674                 return this.compiled(values);
4675             }
4676             var useF = this.disableFormats !== true;
4677             var fm = Roo.util.Format, tpl = this;
4678             var fn = function(m, name, format, args){
4679                 if(format && useF){
4680                     if(format.substr(0, 5) == "this."){
4681                         return tpl.call(format.substr(5), values[name], values);
4682                     }else{
4683                         if(args){
4684                             // quoted values are required for strings in compiled templates, 
4685                             // but for non compiled we need to strip them
4686                             // quoted reversed for jsmin
4687                             var re = /^\s*['"](.*)["']\s*$/;
4688                             args = args.split(',');
4689                             for(var i = 0, len = args.length; i < len; i++){
4690                                 args[i] = args[i].replace(re, "$1");
4691                             }
4692                             args = [values[name]].concat(args);
4693                         }else{
4694                             args = [values[name]];
4695                         }
4696                         return fm[format].apply(fm, args);
4697                     }
4698                 }else{
4699                     return values[name] !== undefined ? values[name] : "";
4700                 }
4701             };
4702             return this.html.replace(this.re, fn);
4703         } catch (e) {
4704             Roo.log(e);
4705             throw e;
4706         }
4707          
4708     },
4709     
4710     loading : false,
4711       
4712     load : function ()
4713     {
4714          
4715         if (this.loading) {
4716             return;
4717         }
4718         var _t = this;
4719         
4720         this.loading = true;
4721         this.compiled = false;
4722         
4723         var cx = new Roo.data.Connection();
4724         cx.request({
4725             url : this.url,
4726             method : 'GET',
4727             success : function (response) {
4728                 _t.loading = false;
4729                 _t.html = response.responseText;
4730                 _t.url = false;
4731                 _t.compile();
4732              },
4733             failure : function(response) {
4734                 Roo.log("Template failed to load from " + _t.url);
4735                 _t.loading = false;
4736             }
4737         });
4738     },
4739
4740     /**
4741      * Sets the HTML used as the template and optionally compiles it.
4742      * @param {String} html
4743      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4744      * @return {Roo.Template} this
4745      */
4746     set : function(html, compile){
4747         this.html = html;
4748         this.compiled = null;
4749         if(compile){
4750             this.compile();
4751         }
4752         return this;
4753     },
4754     
4755     /**
4756      * True to disable format functions (defaults to false)
4757      * @type Boolean
4758      */
4759     disableFormats : false,
4760     
4761     /**
4762     * The regular expression used to match template variables 
4763     * @type RegExp
4764     * @property 
4765     */
4766     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4767     
4768     /**
4769      * Compiles the template into an internal function, eliminating the RegEx overhead.
4770      * @return {Roo.Template} this
4771      */
4772     compile : function(){
4773         var fm = Roo.util.Format;
4774         var useF = this.disableFormats !== true;
4775         var sep = Roo.isGecko ? "+" : ",";
4776         var fn = function(m, name, format, args){
4777             if(format && useF){
4778                 args = args ? ',' + args : "";
4779                 if(format.substr(0, 5) != "this."){
4780                     format = "fm." + format + '(';
4781                 }else{
4782                     format = 'this.call("'+ format.substr(5) + '", ';
4783                     args = ", values";
4784                 }
4785             }else{
4786                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4787             }
4788             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4789         };
4790         var body;
4791         // branched to use + in gecko and [].join() in others
4792         if(Roo.isGecko){
4793             body = "this.compiled = function(values){ return '" +
4794                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4795                     "';};";
4796         }else{
4797             body = ["this.compiled = function(values){ return ['"];
4798             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4799             body.push("'].join('');};");
4800             body = body.join('');
4801         }
4802         /**
4803          * eval:var:values
4804          * eval:var:fm
4805          */
4806         eval(body);
4807         return this;
4808     },
4809     
4810     // private function used to call members
4811     call : function(fnName, value, allValues){
4812         return this[fnName](value, allValues);
4813     },
4814     
4815     /**
4816      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4817      * @param {String/HTMLElement/Roo.Element} el The context element
4818      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4819      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4820      * @return {HTMLElement/Roo.Element} The new node or Element
4821      */
4822     insertFirst: function(el, values, returnElement){
4823         return this.doInsert('afterBegin', el, values, returnElement);
4824     },
4825
4826     /**
4827      * Applies the supplied values to the template and inserts the new node(s) before el.
4828      * @param {String/HTMLElement/Roo.Element} el The context element
4829      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4830      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4831      * @return {HTMLElement/Roo.Element} The new node or Element
4832      */
4833     insertBefore: function(el, values, returnElement){
4834         return this.doInsert('beforeBegin', el, values, returnElement);
4835     },
4836
4837     /**
4838      * Applies the supplied values to the template and inserts the new node(s) after el.
4839      * @param {String/HTMLElement/Roo.Element} el The context element
4840      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4841      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4842      * @return {HTMLElement/Roo.Element} The new node or Element
4843      */
4844     insertAfter : function(el, values, returnElement){
4845         return this.doInsert('afterEnd', el, values, returnElement);
4846     },
4847     
4848     /**
4849      * Applies the supplied values to the template and appends the new node(s) to el.
4850      * @param {String/HTMLElement/Roo.Element} el The context element
4851      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4852      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4853      * @return {HTMLElement/Roo.Element} The new node or Element
4854      */
4855     append : function(el, values, returnElement){
4856         return this.doInsert('beforeEnd', el, values, returnElement);
4857     },
4858
4859     doInsert : function(where, el, values, returnEl){
4860         el = Roo.getDom(el);
4861         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4862         return returnEl ? Roo.get(newNode, true) : newNode;
4863     },
4864
4865     /**
4866      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4867      * @param {String/HTMLElement/Roo.Element} el The context element
4868      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4869      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4870      * @return {HTMLElement/Roo.Element} The new node or Element
4871      */
4872     overwrite : function(el, values, returnElement){
4873         el = Roo.getDom(el);
4874         el.innerHTML = this.applyTemplate(values);
4875         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4876     }
4877 };
4878 /**
4879  * Alias for {@link #applyTemplate}
4880  * @method
4881  */
4882 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4883
4884 // backwards compat
4885 Roo.DomHelper.Template = Roo.Template;
4886
4887 /**
4888  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4889  * @param {String/HTMLElement} el A DOM element or its id
4890  * @returns {Roo.Template} The created template
4891  * @static
4892  */
4893 Roo.Template.from = function(el){
4894     el = Roo.getDom(el);
4895     return new Roo.Template(el.value || el.innerHTML);
4896 };/*
4897  * Based on:
4898  * Ext JS Library 1.1.1
4899  * Copyright(c) 2006-2007, Ext JS, LLC.
4900  *
4901  * Originally Released Under LGPL - original licence link has changed is not relivant.
4902  *
4903  * Fork - LGPL
4904  * <script type="text/javascript">
4905  */
4906  
4907
4908 /*
4909  * This is code is also distributed under MIT license for use
4910  * with jQuery and prototype JavaScript libraries.
4911  */
4912 /**
4913  * @class Roo.DomQuery
4914 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4915 <p>
4916 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4917
4918 <p>
4919 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4920 </p>
4921 <h4>Element Selectors:</h4>
4922 <ul class="list">
4923     <li> <b>*</b> any element</li>
4924     <li> <b>E</b> an element with the tag E</li>
4925     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4926     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4927     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4928     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4929 </ul>
4930 <h4>Attribute Selectors:</h4>
4931 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4932 <ul class="list">
4933     <li> <b>E[foo]</b> has an attribute "foo"</li>
4934     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4935     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4936     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4937     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4938     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4939     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4940 </ul>
4941 <h4>Pseudo Classes:</h4>
4942 <ul class="list">
4943     <li> <b>E:first-child</b> E is the first child of its parent</li>
4944     <li> <b>E:last-child</b> E is the last child of its parent</li>
4945     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4946     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4947     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4948     <li> <b>E:only-child</b> E is the only child of its parent</li>
4949     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4950     <li> <b>E:first</b> the first E in the resultset</li>
4951     <li> <b>E:last</b> the last E in the resultset</li>
4952     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4953     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4954     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4955     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4956     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4957     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4958     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4959     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4960     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4961 </ul>
4962 <h4>CSS Value Selectors:</h4>
4963 <ul class="list">
4964     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4965     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4966     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4967     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4968     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4969     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4970 </ul>
4971  * @singleton
4972  */
4973 Roo.DomQuery = function(){
4974     var cache = {}, simpleCache = {}, valueCache = {};
4975     var nonSpace = /\S/;
4976     var trimRe = /^\s+|\s+$/g;
4977     var tplRe = /\{(\d+)\}/g;
4978     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4979     var tagTokenRe = /^(#)?([\w-\*]+)/;
4980     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4981
4982     function child(p, index){
4983         var i = 0;
4984         var n = p.firstChild;
4985         while(n){
4986             if(n.nodeType == 1){
4987                if(++i == index){
4988                    return n;
4989                }
4990             }
4991             n = n.nextSibling;
4992         }
4993         return null;
4994     };
4995
4996     function next(n){
4997         while((n = n.nextSibling) && n.nodeType != 1);
4998         return n;
4999     };
5000
5001     function prev(n){
5002         while((n = n.previousSibling) && n.nodeType != 1);
5003         return n;
5004     };
5005
5006     function children(d){
5007         var n = d.firstChild, ni = -1;
5008             while(n){
5009                 var nx = n.nextSibling;
5010                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5011                     d.removeChild(n);
5012                 }else{
5013                     n.nodeIndex = ++ni;
5014                 }
5015                 n = nx;
5016             }
5017             return this;
5018         };
5019
5020     function byClassName(c, a, v){
5021         if(!v){
5022             return c;
5023         }
5024         var r = [], ri = -1, cn;
5025         for(var i = 0, ci; ci = c[i]; i++){
5026             if((' '+ci.className+' ').indexOf(v) != -1){
5027                 r[++ri] = ci;
5028             }
5029         }
5030         return r;
5031     };
5032
5033     function attrValue(n, attr){
5034         if(!n.tagName && typeof n.length != "undefined"){
5035             n = n[0];
5036         }
5037         if(!n){
5038             return null;
5039         }
5040         if(attr == "for"){
5041             return n.htmlFor;
5042         }
5043         if(attr == "class" || attr == "className"){
5044             return n.className;
5045         }
5046         return n.getAttribute(attr) || n[attr];
5047
5048     };
5049
5050     function getNodes(ns, mode, tagName){
5051         var result = [], ri = -1, cs;
5052         if(!ns){
5053             return result;
5054         }
5055         tagName = tagName || "*";
5056         if(typeof ns.getElementsByTagName != "undefined"){
5057             ns = [ns];
5058         }
5059         if(!mode){
5060             for(var i = 0, ni; ni = ns[i]; i++){
5061                 cs = ni.getElementsByTagName(tagName);
5062                 for(var j = 0, ci; ci = cs[j]; j++){
5063                     result[++ri] = ci;
5064                 }
5065             }
5066         }else if(mode == "/" || mode == ">"){
5067             var utag = tagName.toUpperCase();
5068             for(var i = 0, ni, cn; ni = ns[i]; i++){
5069                 cn = ni.children || ni.childNodes;
5070                 for(var j = 0, cj; cj = cn[j]; j++){
5071                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5072                         result[++ri] = cj;
5073                     }
5074                 }
5075             }
5076         }else if(mode == "+"){
5077             var utag = tagName.toUpperCase();
5078             for(var i = 0, n; n = ns[i]; i++){
5079                 while((n = n.nextSibling) && n.nodeType != 1);
5080                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5081                     result[++ri] = n;
5082                 }
5083             }
5084         }else if(mode == "~"){
5085             for(var i = 0, n; n = ns[i]; i++){
5086                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5087                 if(n){
5088                     result[++ri] = n;
5089                 }
5090             }
5091         }
5092         return result;
5093     };
5094
5095     function concat(a, b){
5096         if(b.slice){
5097             return a.concat(b);
5098         }
5099         for(var i = 0, l = b.length; i < l; i++){
5100             a[a.length] = b[i];
5101         }
5102         return a;
5103     }
5104
5105     function byTag(cs, tagName){
5106         if(cs.tagName || cs == document){
5107             cs = [cs];
5108         }
5109         if(!tagName){
5110             return cs;
5111         }
5112         var r = [], ri = -1;
5113         tagName = tagName.toLowerCase();
5114         for(var i = 0, ci; ci = cs[i]; i++){
5115             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5116                 r[++ri] = ci;
5117             }
5118         }
5119         return r;
5120     };
5121
5122     function byId(cs, attr, id){
5123         if(cs.tagName || cs == document){
5124             cs = [cs];
5125         }
5126         if(!id){
5127             return cs;
5128         }
5129         var r = [], ri = -1;
5130         for(var i = 0,ci; ci = cs[i]; i++){
5131             if(ci && ci.id == id){
5132                 r[++ri] = ci;
5133                 return r;
5134             }
5135         }
5136         return r;
5137     };
5138
5139     function byAttribute(cs, attr, value, op, custom){
5140         var r = [], ri = -1, st = custom=="{";
5141         var f = Roo.DomQuery.operators[op];
5142         for(var i = 0, ci; ci = cs[i]; i++){
5143             var a;
5144             if(st){
5145                 a = Roo.DomQuery.getStyle(ci, attr);
5146             }
5147             else if(attr == "class" || attr == "className"){
5148                 a = ci.className;
5149             }else if(attr == "for"){
5150                 a = ci.htmlFor;
5151             }else if(attr == "href"){
5152                 a = ci.getAttribute("href", 2);
5153             }else{
5154                 a = ci.getAttribute(attr);
5155             }
5156             if((f && f(a, value)) || (!f && a)){
5157                 r[++ri] = ci;
5158             }
5159         }
5160         return r;
5161     };
5162
5163     function byPseudo(cs, name, value){
5164         return Roo.DomQuery.pseudos[name](cs, value);
5165     };
5166
5167     // This is for IE MSXML which does not support expandos.
5168     // IE runs the same speed using setAttribute, however FF slows way down
5169     // and Safari completely fails so they need to continue to use expandos.
5170     var isIE = window.ActiveXObject ? true : false;
5171
5172     // this eval is stop the compressor from
5173     // renaming the variable to something shorter
5174     
5175     /** eval:var:batch */
5176     var batch = 30803; 
5177
5178     var key = 30803;
5179
5180     function nodupIEXml(cs){
5181         var d = ++key;
5182         cs[0].setAttribute("_nodup", d);
5183         var r = [cs[0]];
5184         for(var i = 1, len = cs.length; i < len; i++){
5185             var c = cs[i];
5186             if(!c.getAttribute("_nodup") != d){
5187                 c.setAttribute("_nodup", d);
5188                 r[r.length] = c;
5189             }
5190         }
5191         for(var i = 0, len = cs.length; i < len; i++){
5192             cs[i].removeAttribute("_nodup");
5193         }
5194         return r;
5195     }
5196
5197     function nodup(cs){
5198         if(!cs){
5199             return [];
5200         }
5201         var len = cs.length, c, i, r = cs, cj, ri = -1;
5202         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5203             return cs;
5204         }
5205         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5206             return nodupIEXml(cs);
5207         }
5208         var d = ++key;
5209         cs[0]._nodup = d;
5210         for(i = 1; c = cs[i]; i++){
5211             if(c._nodup != d){
5212                 c._nodup = d;
5213             }else{
5214                 r = [];
5215                 for(var j = 0; j < i; j++){
5216                     r[++ri] = cs[j];
5217                 }
5218                 for(j = i+1; cj = cs[j]; j++){
5219                     if(cj._nodup != d){
5220                         cj._nodup = d;
5221                         r[++ri] = cj;
5222                     }
5223                 }
5224                 return r;
5225             }
5226         }
5227         return r;
5228     }
5229
5230     function quickDiffIEXml(c1, c2){
5231         var d = ++key;
5232         for(var i = 0, len = c1.length; i < len; i++){
5233             c1[i].setAttribute("_qdiff", d);
5234         }
5235         var r = [];
5236         for(var i = 0, len = c2.length; i < len; i++){
5237             if(c2[i].getAttribute("_qdiff") != d){
5238                 r[r.length] = c2[i];
5239             }
5240         }
5241         for(var i = 0, len = c1.length; i < len; i++){
5242            c1[i].removeAttribute("_qdiff");
5243         }
5244         return r;
5245     }
5246
5247     function quickDiff(c1, c2){
5248         var len1 = c1.length;
5249         if(!len1){
5250             return c2;
5251         }
5252         if(isIE && c1[0].selectSingleNode){
5253             return quickDiffIEXml(c1, c2);
5254         }
5255         var d = ++key;
5256         for(var i = 0; i < len1; i++){
5257             c1[i]._qdiff = d;
5258         }
5259         var r = [];
5260         for(var i = 0, len = c2.length; i < len; i++){
5261             if(c2[i]._qdiff != d){
5262                 r[r.length] = c2[i];
5263             }
5264         }
5265         return r;
5266     }
5267
5268     function quickId(ns, mode, root, id){
5269         if(ns == root){
5270            var d = root.ownerDocument || root;
5271            return d.getElementById(id);
5272         }
5273         ns = getNodes(ns, mode, "*");
5274         return byId(ns, null, id);
5275     }
5276
5277     return {
5278         getStyle : function(el, name){
5279             return Roo.fly(el).getStyle(name);
5280         },
5281         /**
5282          * Compiles a selector/xpath query into a reusable function. The returned function
5283          * takes one parameter "root" (optional), which is the context node from where the query should start.
5284          * @param {String} selector The selector/xpath query
5285          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5286          * @return {Function}
5287          */
5288         compile : function(path, type){
5289             type = type || "select";
5290             
5291             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5292             var q = path, mode, lq;
5293             var tk = Roo.DomQuery.matchers;
5294             var tklen = tk.length;
5295             var mm;
5296
5297             // accept leading mode switch
5298             var lmode = q.match(modeRe);
5299             if(lmode && lmode[1]){
5300                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5301                 q = q.replace(lmode[1], "");
5302             }
5303             // strip leading slashes
5304             while(path.substr(0, 1)=="/"){
5305                 path = path.substr(1);
5306             }
5307
5308             while(q && lq != q){
5309                 lq = q;
5310                 var tm = q.match(tagTokenRe);
5311                 if(type == "select"){
5312                     if(tm){
5313                         if(tm[1] == "#"){
5314                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5315                         }else{
5316                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5317                         }
5318                         q = q.replace(tm[0], "");
5319                     }else if(q.substr(0, 1) != '@'){
5320                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5321                     }
5322                 }else{
5323                     if(tm){
5324                         if(tm[1] == "#"){
5325                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5326                         }else{
5327                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5328                         }
5329                         q = q.replace(tm[0], "");
5330                     }
5331                 }
5332                 while(!(mm = q.match(modeRe))){
5333                     var matched = false;
5334                     for(var j = 0; j < tklen; j++){
5335                         var t = tk[j];
5336                         var m = q.match(t.re);
5337                         if(m){
5338                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5339                                                     return m[i];
5340                                                 });
5341                             q = q.replace(m[0], "");
5342                             matched = true;
5343                             break;
5344                         }
5345                     }
5346                     // prevent infinite loop on bad selector
5347                     if(!matched){
5348                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5349                     }
5350                 }
5351                 if(mm[1]){
5352                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5353                     q = q.replace(mm[1], "");
5354                 }
5355             }
5356             fn[fn.length] = "return nodup(n);\n}";
5357             
5358              /** 
5359               * list of variables that need from compression as they are used by eval.
5360              *  eval:var:batch 
5361              *  eval:var:nodup
5362              *  eval:var:byTag
5363              *  eval:var:ById
5364              *  eval:var:getNodes
5365              *  eval:var:quickId
5366              *  eval:var:mode
5367              *  eval:var:root
5368              *  eval:var:n
5369              *  eval:var:byClassName
5370              *  eval:var:byPseudo
5371              *  eval:var:byAttribute
5372              *  eval:var:attrValue
5373              * 
5374              **/ 
5375             eval(fn.join(""));
5376             return f;
5377         },
5378
5379         /**
5380          * Selects a group of elements.
5381          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5382          * @param {Node} root (optional) The start of the query (defaults to document).
5383          * @return {Array}
5384          */
5385         select : function(path, root, type){
5386             if(!root || root == document){
5387                 root = document;
5388             }
5389             if(typeof root == "string"){
5390                 root = document.getElementById(root);
5391             }
5392             var paths = path.split(",");
5393             var results = [];
5394             for(var i = 0, len = paths.length; i < len; i++){
5395                 var p = paths[i].replace(trimRe, "");
5396                 if(!cache[p]){
5397                     cache[p] = Roo.DomQuery.compile(p);
5398                     if(!cache[p]){
5399                         throw p + " is not a valid selector";
5400                     }
5401                 }
5402                 var result = cache[p](root);
5403                 if(result && result != document){
5404                     results = results.concat(result);
5405                 }
5406             }
5407             if(paths.length > 1){
5408                 return nodup(results);
5409             }
5410             return results;
5411         },
5412
5413         /**
5414          * Selects a single element.
5415          * @param {String} selector The selector/xpath query
5416          * @param {Node} root (optional) The start of the query (defaults to document).
5417          * @return {Element}
5418          */
5419         selectNode : function(path, root){
5420             return Roo.DomQuery.select(path, root)[0];
5421         },
5422
5423         /**
5424          * Selects the value of a node, optionally replacing null with the defaultValue.
5425          * @param {String} selector The selector/xpath query
5426          * @param {Node} root (optional) The start of the query (defaults to document).
5427          * @param {String} defaultValue
5428          */
5429         selectValue : function(path, root, defaultValue){
5430             path = path.replace(trimRe, "");
5431             if(!valueCache[path]){
5432                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5433             }
5434             var n = valueCache[path](root);
5435             n = n[0] ? n[0] : n;
5436             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5437             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5438         },
5439
5440         /**
5441          * Selects the value of a node, parsing integers and floats.
5442          * @param {String} selector The selector/xpath query
5443          * @param {Node} root (optional) The start of the query (defaults to document).
5444          * @param {Number} defaultValue
5445          * @return {Number}
5446          */
5447         selectNumber : function(path, root, defaultValue){
5448             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5449             return parseFloat(v);
5450         },
5451
5452         /**
5453          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5454          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5455          * @param {String} selector The simple selector to test
5456          * @return {Boolean}
5457          */
5458         is : function(el, ss){
5459             if(typeof el == "string"){
5460                 el = document.getElementById(el);
5461             }
5462             var isArray = (el instanceof Array);
5463             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5464             return isArray ? (result.length == el.length) : (result.length > 0);
5465         },
5466
5467         /**
5468          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5469          * @param {Array} el An array of elements to filter
5470          * @param {String} selector The simple selector to test
5471          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5472          * the selector instead of the ones that match
5473          * @return {Array}
5474          */
5475         filter : function(els, ss, nonMatches){
5476             ss = ss.replace(trimRe, "");
5477             if(!simpleCache[ss]){
5478                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5479             }
5480             var result = simpleCache[ss](els);
5481             return nonMatches ? quickDiff(result, els) : result;
5482         },
5483
5484         /**
5485          * Collection of matching regular expressions and code snippets.
5486          */
5487         matchers : [{
5488                 re: /^\.([\w-]+)/,
5489                 select: 'n = byClassName(n, null, " {1} ");'
5490             }, {
5491                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5492                 select: 'n = byPseudo(n, "{1}", "{2}");'
5493             },{
5494                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5495                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5496             }, {
5497                 re: /^#([\w-]+)/,
5498                 select: 'n = byId(n, null, "{1}");'
5499             },{
5500                 re: /^@([\w-]+)/,
5501                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5502             }
5503         ],
5504
5505         /**
5506          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5507          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5508          */
5509         operators : {
5510             "=" : function(a, v){
5511                 return a == v;
5512             },
5513             "!=" : function(a, v){
5514                 return a != v;
5515             },
5516             "^=" : function(a, v){
5517                 return a && a.substr(0, v.length) == v;
5518             },
5519             "$=" : function(a, v){
5520                 return a && a.substr(a.length-v.length) == v;
5521             },
5522             "*=" : function(a, v){
5523                 return a && a.indexOf(v) !== -1;
5524             },
5525             "%=" : function(a, v){
5526                 return (a % v) == 0;
5527             },
5528             "|=" : function(a, v){
5529                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5530             },
5531             "~=" : function(a, v){
5532                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5533             }
5534         },
5535
5536         /**
5537          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5538          * and the argument (if any) supplied in the selector.
5539          */
5540         pseudos : {
5541             "first-child" : function(c){
5542                 var r = [], ri = -1, n;
5543                 for(var i = 0, ci; ci = n = c[i]; i++){
5544                     while((n = n.previousSibling) && n.nodeType != 1);
5545                     if(!n){
5546                         r[++ri] = ci;
5547                     }
5548                 }
5549                 return r;
5550             },
5551
5552             "last-child" : function(c){
5553                 var r = [], ri = -1, n;
5554                 for(var i = 0, ci; ci = n = c[i]; i++){
5555                     while((n = n.nextSibling) && n.nodeType != 1);
5556                     if(!n){
5557                         r[++ri] = ci;
5558                     }
5559                 }
5560                 return r;
5561             },
5562
5563             "nth-child" : function(c, a) {
5564                 var r = [], ri = -1;
5565                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5566                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5567                 for(var i = 0, n; n = c[i]; i++){
5568                     var pn = n.parentNode;
5569                     if (batch != pn._batch) {
5570                         var j = 0;
5571                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5572                             if(cn.nodeType == 1){
5573                                cn.nodeIndex = ++j;
5574                             }
5575                         }
5576                         pn._batch = batch;
5577                     }
5578                     if (f == 1) {
5579                         if (l == 0 || n.nodeIndex == l){
5580                             r[++ri] = n;
5581                         }
5582                     } else if ((n.nodeIndex + l) % f == 0){
5583                         r[++ri] = n;
5584                     }
5585                 }
5586
5587                 return r;
5588             },
5589
5590             "only-child" : function(c){
5591                 var r = [], ri = -1;;
5592                 for(var i = 0, ci; ci = c[i]; i++){
5593                     if(!prev(ci) && !next(ci)){
5594                         r[++ri] = ci;
5595                     }
5596                 }
5597                 return r;
5598             },
5599
5600             "empty" : function(c){
5601                 var r = [], ri = -1;
5602                 for(var i = 0, ci; ci = c[i]; i++){
5603                     var cns = ci.childNodes, j = 0, cn, empty = true;
5604                     while(cn = cns[j]){
5605                         ++j;
5606                         if(cn.nodeType == 1 || cn.nodeType == 3){
5607                             empty = false;
5608                             break;
5609                         }
5610                     }
5611                     if(empty){
5612                         r[++ri] = ci;
5613                     }
5614                 }
5615                 return r;
5616             },
5617
5618             "contains" : function(c, v){
5619                 var r = [], ri = -1;
5620                 for(var i = 0, ci; ci = c[i]; i++){
5621                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5622                         r[++ri] = ci;
5623                     }
5624                 }
5625                 return r;
5626             },
5627
5628             "nodeValue" : function(c, v){
5629                 var r = [], ri = -1;
5630                 for(var i = 0, ci; ci = c[i]; i++){
5631                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5632                         r[++ri] = ci;
5633                     }
5634                 }
5635                 return r;
5636             },
5637
5638             "checked" : function(c){
5639                 var r = [], ri = -1;
5640                 for(var i = 0, ci; ci = c[i]; i++){
5641                     if(ci.checked == true){
5642                         r[++ri] = ci;
5643                     }
5644                 }
5645                 return r;
5646             },
5647
5648             "not" : function(c, ss){
5649                 return Roo.DomQuery.filter(c, ss, true);
5650             },
5651
5652             "odd" : function(c){
5653                 return this["nth-child"](c, "odd");
5654             },
5655
5656             "even" : function(c){
5657                 return this["nth-child"](c, "even");
5658             },
5659
5660             "nth" : function(c, a){
5661                 return c[a-1] || [];
5662             },
5663
5664             "first" : function(c){
5665                 return c[0] || [];
5666             },
5667
5668             "last" : function(c){
5669                 return c[c.length-1] || [];
5670             },
5671
5672             "has" : function(c, ss){
5673                 var s = Roo.DomQuery.select;
5674                 var r = [], ri = -1;
5675                 for(var i = 0, ci; ci = c[i]; i++){
5676                     if(s(ss, ci).length > 0){
5677                         r[++ri] = ci;
5678                     }
5679                 }
5680                 return r;
5681             },
5682
5683             "next" : function(c, ss){
5684                 var is = Roo.DomQuery.is;
5685                 var r = [], ri = -1;
5686                 for(var i = 0, ci; ci = c[i]; i++){
5687                     var n = next(ci);
5688                     if(n && is(n, ss)){
5689                         r[++ri] = ci;
5690                     }
5691                 }
5692                 return r;
5693             },
5694
5695             "prev" : function(c, ss){
5696                 var is = Roo.DomQuery.is;
5697                 var r = [], ri = -1;
5698                 for(var i = 0, ci; ci = c[i]; i++){
5699                     var n = prev(ci);
5700                     if(n && is(n, ss)){
5701                         r[++ri] = ci;
5702                     }
5703                 }
5704                 return r;
5705             }
5706         }
5707     };
5708 }();
5709
5710 /**
5711  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5712  * @param {String} path The selector/xpath query
5713  * @param {Node} root (optional) The start of the query (defaults to document).
5714  * @return {Array}
5715  * @member Roo
5716  * @method query
5717  */
5718 Roo.query = Roo.DomQuery.select;
5719 /*
5720  * Based on:
5721  * Ext JS Library 1.1.1
5722  * Copyright(c) 2006-2007, Ext JS, LLC.
5723  *
5724  * Originally Released Under LGPL - original licence link has changed is not relivant.
5725  *
5726  * Fork - LGPL
5727  * <script type="text/javascript">
5728  */
5729
5730 /**
5731  * @class Roo.util.Observable
5732  * Base class that provides a common interface for publishing events. Subclasses are expected to
5733  * to have a property "events" with all the events defined.<br>
5734  * For example:
5735  * <pre><code>
5736  Employee = function(name){
5737     this.name = name;
5738     this.addEvents({
5739         "fired" : true,
5740         "quit" : true
5741     });
5742  }
5743  Roo.extend(Employee, Roo.util.Observable);
5744 </code></pre>
5745  * @param {Object} config properties to use (incuding events / listeners)
5746  */
5747
5748 Roo.util.Observable = function(cfg){
5749     
5750     cfg = cfg|| {};
5751     this.addEvents(cfg.events || {});
5752     if (cfg.events) {
5753         delete cfg.events; // make sure
5754     }
5755      
5756     Roo.apply(this, cfg);
5757     
5758     if(this.listeners){
5759         this.on(this.listeners);
5760         delete this.listeners;
5761     }
5762 };
5763 Roo.util.Observable.prototype = {
5764     /** 
5765  * @cfg {Object} listeners  list of events and functions to call for this object, 
5766  * For example :
5767  * <pre><code>
5768     listeners :  { 
5769        'click' : function(e) {
5770            ..... 
5771         } ,
5772         .... 
5773     } 
5774   </code></pre>
5775  */
5776     
5777     
5778     /**
5779      * Fires the specified event with the passed parameters (minus the event name).
5780      * @param {String} eventName
5781      * @param {Object...} args Variable number of parameters are passed to handlers
5782      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5783      */
5784     fireEvent : function(){
5785         var ce = this.events[arguments[0].toLowerCase()];
5786         if(typeof ce == "object"){
5787             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5788         }else{
5789             return true;
5790         }
5791     },
5792
5793     // private
5794     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5795
5796     /**
5797      * Appends an event handler to this component
5798      * @param {String}   eventName The type of event to listen for
5799      * @param {Function} handler The method the event invokes
5800      * @param {Object}   scope (optional) The scope in which to execute the handler
5801      * function. The handler function's "this" context.
5802      * @param {Object}   options (optional) An object containing handler configuration
5803      * properties. This may contain any of the following properties:<ul>
5804      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5805      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5806      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5807      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5808      * by the specified number of milliseconds. If the event fires again within that time, the original
5809      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5810      * </ul><br>
5811      * <p>
5812      * <b>Combining Options</b><br>
5813      * Using the options argument, it is possible to combine different types of listeners:<br>
5814      * <br>
5815      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5816                 <pre><code>
5817                 el.on('click', this.onClick, this, {
5818                         single: true,
5819                 delay: 100,
5820                 forumId: 4
5821                 });
5822                 </code></pre>
5823      * <p>
5824      * <b>Attaching multiple handlers in 1 call</b><br>
5825      * The method also allows for a single argument to be passed which is a config object containing properties
5826      * which specify multiple handlers.
5827      * <pre><code>
5828                 el.on({
5829                         'click': {
5830                         fn: this.onClick,
5831                         scope: this,
5832                         delay: 100
5833                 }, 
5834                 'mouseover': {
5835                         fn: this.onMouseOver,
5836                         scope: this
5837                 },
5838                 'mouseout': {
5839                         fn: this.onMouseOut,
5840                         scope: this
5841                 }
5842                 });
5843                 </code></pre>
5844      * <p>
5845      * Or a shorthand syntax which passes the same scope object to all handlers:
5846         <pre><code>
5847                 el.on({
5848                         'click': this.onClick,
5849                 'mouseover': this.onMouseOver,
5850                 'mouseout': this.onMouseOut,
5851                 scope: this
5852                 });
5853                 </code></pre>
5854      */
5855     addListener : function(eventName, fn, scope, o){
5856         if(typeof eventName == "object"){
5857             o = eventName;
5858             for(var e in o){
5859                 if(this.filterOptRe.test(e)){
5860                     continue;
5861                 }
5862                 if(typeof o[e] == "function"){
5863                     // shared options
5864                     this.addListener(e, o[e], o.scope,  o);
5865                 }else{
5866                     // individual options
5867                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5868                 }
5869             }
5870             return;
5871         }
5872         o = (!o || typeof o == "boolean") ? {} : o;
5873         eventName = eventName.toLowerCase();
5874         var ce = this.events[eventName] || true;
5875         if(typeof ce == "boolean"){
5876             ce = new Roo.util.Event(this, eventName);
5877             this.events[eventName] = ce;
5878         }
5879         ce.addListener(fn, scope, o);
5880     },
5881
5882     /**
5883      * Removes a listener
5884      * @param {String}   eventName     The type of event to listen for
5885      * @param {Function} handler        The handler to remove
5886      * @param {Object}   scope  (optional) The scope (this object) for the handler
5887      */
5888     removeListener : function(eventName, fn, scope){
5889         var ce = this.events[eventName.toLowerCase()];
5890         if(typeof ce == "object"){
5891             ce.removeListener(fn, scope);
5892         }
5893     },
5894
5895     /**
5896      * Removes all listeners for this object
5897      */
5898     purgeListeners : function(){
5899         for(var evt in this.events){
5900             if(typeof this.events[evt] == "object"){
5901                  this.events[evt].clearListeners();
5902             }
5903         }
5904     },
5905
5906     relayEvents : function(o, events){
5907         var createHandler = function(ename){
5908             return function(){
5909                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5910             };
5911         };
5912         for(var i = 0, len = events.length; i < len; i++){
5913             var ename = events[i];
5914             if(!this.events[ename]){ this.events[ename] = true; };
5915             o.on(ename, createHandler(ename), this);
5916         }
5917     },
5918
5919     /**
5920      * Used to define events on this Observable
5921      * @param {Object} object The object with the events defined
5922      */
5923     addEvents : function(o){
5924         if(!this.events){
5925             this.events = {};
5926         }
5927         Roo.applyIf(this.events, o);
5928     },
5929
5930     /**
5931      * Checks to see if this object has any listeners for a specified event
5932      * @param {String} eventName The name of the event to check for
5933      * @return {Boolean} True if the event is being listened for, else false
5934      */
5935     hasListener : function(eventName){
5936         var e = this.events[eventName];
5937         return typeof e == "object" && e.listeners.length > 0;
5938     }
5939 };
5940 /**
5941  * Appends an event handler to this element (shorthand for addListener)
5942  * @param {String}   eventName     The type of event to listen for
5943  * @param {Function} handler        The method the event invokes
5944  * @param {Object}   scope (optional) The scope in which to execute the handler
5945  * function. The handler function's "this" context.
5946  * @param {Object}   options  (optional)
5947  * @method
5948  */
5949 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5950 /**
5951  * Removes a listener (shorthand for removeListener)
5952  * @param {String}   eventName     The type of event to listen for
5953  * @param {Function} handler        The handler to remove
5954  * @param {Object}   scope  (optional) The scope (this object) for the handler
5955  * @method
5956  */
5957 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5958
5959 /**
5960  * Starts capture on the specified Observable. All events will be passed
5961  * to the supplied function with the event name + standard signature of the event
5962  * <b>before</b> the event is fired. If the supplied function returns false,
5963  * the event will not fire.
5964  * @param {Observable} o The Observable to capture
5965  * @param {Function} fn The function to call
5966  * @param {Object} scope (optional) The scope (this object) for the fn
5967  * @static
5968  */
5969 Roo.util.Observable.capture = function(o, fn, scope){
5970     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5971 };
5972
5973 /**
5974  * Removes <b>all</b> added captures from the Observable.
5975  * @param {Observable} o The Observable to release
5976  * @static
5977  */
5978 Roo.util.Observable.releaseCapture = function(o){
5979     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5980 };
5981
5982 (function(){
5983
5984     var createBuffered = function(h, o, scope){
5985         var task = new Roo.util.DelayedTask();
5986         return function(){
5987             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5988         };
5989     };
5990
5991     var createSingle = function(h, e, fn, scope){
5992         return function(){
5993             e.removeListener(fn, scope);
5994             return h.apply(scope, arguments);
5995         };
5996     };
5997
5998     var createDelayed = function(h, o, scope){
5999         return function(){
6000             var args = Array.prototype.slice.call(arguments, 0);
6001             setTimeout(function(){
6002                 h.apply(scope, args);
6003             }, o.delay || 10);
6004         };
6005     };
6006
6007     Roo.util.Event = function(obj, name){
6008         this.name = name;
6009         this.obj = obj;
6010         this.listeners = [];
6011     };
6012
6013     Roo.util.Event.prototype = {
6014         addListener : function(fn, scope, options){
6015             var o = options || {};
6016             scope = scope || this.obj;
6017             if(!this.isListening(fn, scope)){
6018                 var l = {fn: fn, scope: scope, options: o};
6019                 var h = fn;
6020                 if(o.delay){
6021                     h = createDelayed(h, o, scope);
6022                 }
6023                 if(o.single){
6024                     h = createSingle(h, this, fn, scope);
6025                 }
6026                 if(o.buffer){
6027                     h = createBuffered(h, o, scope);
6028                 }
6029                 l.fireFn = h;
6030                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6031                     this.listeners.push(l);
6032                 }else{
6033                     this.listeners = this.listeners.slice(0);
6034                     this.listeners.push(l);
6035                 }
6036             }
6037         },
6038
6039         findListener : function(fn, scope){
6040             scope = scope || this.obj;
6041             var ls = this.listeners;
6042             for(var i = 0, len = ls.length; i < len; i++){
6043                 var l = ls[i];
6044                 if(l.fn == fn && l.scope == scope){
6045                     return i;
6046                 }
6047             }
6048             return -1;
6049         },
6050
6051         isListening : function(fn, scope){
6052             return this.findListener(fn, scope) != -1;
6053         },
6054
6055         removeListener : function(fn, scope){
6056             var index;
6057             if((index = this.findListener(fn, scope)) != -1){
6058                 if(!this.firing){
6059                     this.listeners.splice(index, 1);
6060                 }else{
6061                     this.listeners = this.listeners.slice(0);
6062                     this.listeners.splice(index, 1);
6063                 }
6064                 return true;
6065             }
6066             return false;
6067         },
6068
6069         clearListeners : function(){
6070             this.listeners = [];
6071         },
6072
6073         fire : function(){
6074             var ls = this.listeners, scope, len = ls.length;
6075             if(len > 0){
6076                 this.firing = true;
6077                 
6078                 for(var i = 0; i < len; i++){
6079                     var args = Array.prototype.slice.call(arguments, 0);
6080                     var l = ls[i];
6081                     args.push(l.options);
6082                     if(l.fireFn.apply(l.scope||this.obj||window, args) === false){
6083                         this.firing = false;
6084                         return false;
6085                     }
6086                 }
6087                 this.firing = false;
6088             }
6089             return true;
6090         }
6091     };
6092 })();/*
6093  * RooJS Library 
6094  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6095  *
6096  * Licence LGPL 
6097  *
6098  */
6099  
6100 /**
6101  * @class Roo.Document
6102  * @extends Roo.util.Observable
6103  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6104  * 
6105  * @param {Object} config the methods and properties of the 'base' class for the application.
6106  * 
6107  *  Generic Page handler - implement this to start your app..
6108  * 
6109  * eg.
6110  *  MyProject = new Roo.Document({
6111         events : {
6112             'load' : true // your events..
6113         },
6114         listeners : {
6115             'ready' : function() {
6116                 // fired on Roo.onReady()
6117             }
6118         }
6119  * 
6120  */
6121 Roo.Document = function(cfg) {
6122      
6123     this.addEvents({ 
6124         'ready' : true
6125     });
6126     Roo.util.Observable.call(this,cfg);
6127     
6128     var _this = this;
6129     
6130     Roo.onReady(function() {
6131         _this.fireEvent('ready');
6132     },null,false);
6133     
6134     
6135 }
6136
6137 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6138  * Based on:
6139  * Ext JS Library 1.1.1
6140  * Copyright(c) 2006-2007, Ext JS, LLC.
6141  *
6142  * Originally Released Under LGPL - original licence link has changed is not relivant.
6143  *
6144  * Fork - LGPL
6145  * <script type="text/javascript">
6146  */
6147
6148 /**
6149  * @class Roo.EventManager
6150  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6151  * several useful events directly.
6152  * See {@link Roo.EventObject} for more details on normalized event objects.
6153  * @singleton
6154  */
6155 Roo.EventManager = function(){
6156     var docReadyEvent, docReadyProcId, docReadyState = false;
6157     var resizeEvent, resizeTask, textEvent, textSize;
6158     var E = Roo.lib.Event;
6159     var D = Roo.lib.Dom;
6160
6161     
6162     
6163
6164     var fireDocReady = function(){
6165         if(!docReadyState){
6166             docReadyState = true;
6167             Roo.isReady = true;
6168             if(docReadyProcId){
6169                 clearInterval(docReadyProcId);
6170             }
6171             if(Roo.isGecko || Roo.isOpera) {
6172                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6173             }
6174             if(Roo.isIE){
6175                 var defer = document.getElementById("ie-deferred-loader");
6176                 if(defer){
6177                     defer.onreadystatechange = null;
6178                     defer.parentNode.removeChild(defer);
6179                 }
6180             }
6181             if(docReadyEvent){
6182                 docReadyEvent.fire();
6183                 docReadyEvent.clearListeners();
6184             }
6185         }
6186     };
6187     
6188     var initDocReady = function(){
6189         docReadyEvent = new Roo.util.Event();
6190         if(Roo.isGecko || Roo.isOpera) {
6191             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6192         }else if(Roo.isIE){
6193             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6194             var defer = document.getElementById("ie-deferred-loader");
6195             defer.onreadystatechange = function(){
6196                 if(this.readyState == "complete"){
6197                     fireDocReady();
6198                 }
6199             };
6200         }else if(Roo.isSafari){ 
6201             docReadyProcId = setInterval(function(){
6202                 var rs = document.readyState;
6203                 if(rs == "complete") {
6204                     fireDocReady();     
6205                  }
6206             }, 10);
6207         }
6208         // no matter what, make sure it fires on load
6209         E.on(window, "load", fireDocReady);
6210     };
6211
6212     var createBuffered = function(h, o){
6213         var task = new Roo.util.DelayedTask(h);
6214         return function(e){
6215             // create new event object impl so new events don't wipe out properties
6216             e = new Roo.EventObjectImpl(e);
6217             task.delay(o.buffer, h, null, [e]);
6218         };
6219     };
6220
6221     var createSingle = function(h, el, ename, fn){
6222         return function(e){
6223             Roo.EventManager.removeListener(el, ename, fn);
6224             h(e);
6225         };
6226     };
6227
6228     var createDelayed = function(h, o){
6229         return function(e){
6230             // create new event object impl so new events don't wipe out properties
6231             e = new Roo.EventObjectImpl(e);
6232             setTimeout(function(){
6233                 h(e);
6234             }, o.delay || 10);
6235         };
6236     };
6237     var transitionEndVal = false;
6238     
6239     var transitionEnd = function()
6240     {
6241         if (transitionEndVal) {
6242             return transitionEndVal;
6243         }
6244         var el = document.createElement('div');
6245
6246         var transEndEventNames = {
6247             WebkitTransition : 'webkitTransitionEnd',
6248             MozTransition    : 'transitionend',
6249             OTransition      : 'oTransitionEnd otransitionend',
6250             transition       : 'transitionend'
6251         };
6252     
6253         for (var name in transEndEventNames) {
6254             if (el.style[name] !== undefined) {
6255                 transitionEndVal = transEndEventNames[name];
6256                 return  transitionEndVal ;
6257             }
6258         }
6259     }
6260     
6261
6262     var listen = function(element, ename, opt, fn, scope){
6263         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6264         fn = fn || o.fn; scope = scope || o.scope;
6265         var el = Roo.getDom(element);
6266         
6267         
6268         if(!el){
6269             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6270         }
6271         
6272         if (ename == 'transitionend') {
6273             ename = transitionEnd();
6274         }
6275         var h = function(e){
6276             e = Roo.EventObject.setEvent(e);
6277             var t;
6278             if(o.delegate){
6279                 t = e.getTarget(o.delegate, el);
6280                 if(!t){
6281                     return;
6282                 }
6283             }else{
6284                 t = e.target;
6285             }
6286             if(o.stopEvent === true){
6287                 e.stopEvent();
6288             }
6289             if(o.preventDefault === true){
6290                e.preventDefault();
6291             }
6292             if(o.stopPropagation === true){
6293                 e.stopPropagation();
6294             }
6295
6296             if(o.normalized === false){
6297                 e = e.browserEvent;
6298             }
6299
6300             fn.call(scope || el, e, t, o);
6301         };
6302         if(o.delay){
6303             h = createDelayed(h, o);
6304         }
6305         if(o.single){
6306             h = createSingle(h, el, ename, fn);
6307         }
6308         if(o.buffer){
6309             h = createBuffered(h, o);
6310         }
6311         
6312         fn._handlers = fn._handlers || [];
6313         
6314         
6315         fn._handlers.push([Roo.id(el), ename, h]);
6316         
6317         
6318          
6319         E.on(el, ename, h);
6320         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6321             el.addEventListener("DOMMouseScroll", h, false);
6322             E.on(window, 'unload', function(){
6323                 el.removeEventListener("DOMMouseScroll", h, false);
6324             });
6325         }
6326         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6327             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6328         }
6329         return h;
6330     };
6331
6332     var stopListening = function(el, ename, fn){
6333         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6334         if(hds){
6335             for(var i = 0, len = hds.length; i < len; i++){
6336                 var h = hds[i];
6337                 if(h[0] == id && h[1] == ename){
6338                     hd = h[2];
6339                     hds.splice(i, 1);
6340                     break;
6341                 }
6342             }
6343         }
6344         E.un(el, ename, hd);
6345         el = Roo.getDom(el);
6346         if(ename == "mousewheel" && el.addEventListener){
6347             el.removeEventListener("DOMMouseScroll", hd, false);
6348         }
6349         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6350             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6351         }
6352     };
6353
6354     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6355     
6356     var pub = {
6357         
6358         
6359         /** 
6360          * Fix for doc tools
6361          * @scope Roo.EventManager
6362          */
6363         
6364         
6365         /** 
6366          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6367          * object with a Roo.EventObject
6368          * @param {Function} fn        The method the event invokes
6369          * @param {Object}   scope    An object that becomes the scope of the handler
6370          * @param {boolean}  override If true, the obj passed in becomes
6371          *                             the execution scope of the listener
6372          * @return {Function} The wrapped function
6373          * @deprecated
6374          */
6375         wrap : function(fn, scope, override){
6376             return function(e){
6377                 Roo.EventObject.setEvent(e);
6378                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6379             };
6380         },
6381         
6382         /**
6383      * Appends an event handler to an element (shorthand for addListener)
6384      * @param {String/HTMLElement}   element        The html element or id to assign the
6385      * @param {String}   eventName The type of event to listen for
6386      * @param {Function} handler The method the event invokes
6387      * @param {Object}   scope (optional) The scope in which to execute the handler
6388      * function. The handler function's "this" context.
6389      * @param {Object}   options (optional) An object containing handler configuration
6390      * properties. This may contain any of the following properties:<ul>
6391      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6392      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6393      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6394      * <li>preventDefault {Boolean} True to prevent the default action</li>
6395      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6396      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6397      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6398      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6399      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6400      * by the specified number of milliseconds. If the event fires again within that time, the original
6401      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6402      * </ul><br>
6403      * <p>
6404      * <b>Combining Options</b><br>
6405      * Using the options argument, it is possible to combine different types of listeners:<br>
6406      * <br>
6407      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6408      * Code:<pre><code>
6409 el.on('click', this.onClick, this, {
6410     single: true,
6411     delay: 100,
6412     stopEvent : true,
6413     forumId: 4
6414 });</code></pre>
6415      * <p>
6416      * <b>Attaching multiple handlers in 1 call</b><br>
6417       * The method also allows for a single argument to be passed which is a config object containing properties
6418      * which specify multiple handlers.
6419      * <p>
6420      * Code:<pre><code>
6421 el.on({
6422     'click' : {
6423         fn: this.onClick
6424         scope: this,
6425         delay: 100
6426     },
6427     'mouseover' : {
6428         fn: this.onMouseOver
6429         scope: this
6430     },
6431     'mouseout' : {
6432         fn: this.onMouseOut
6433         scope: this
6434     }
6435 });</code></pre>
6436      * <p>
6437      * Or a shorthand syntax:<br>
6438      * Code:<pre><code>
6439 el.on({
6440     'click' : this.onClick,
6441     'mouseover' : this.onMouseOver,
6442     'mouseout' : this.onMouseOut
6443     scope: this
6444 });</code></pre>
6445      */
6446         addListener : function(element, eventName, fn, scope, options){
6447             if(typeof eventName == "object"){
6448                 var o = eventName;
6449                 for(var e in o){
6450                     if(propRe.test(e)){
6451                         continue;
6452                     }
6453                     if(typeof o[e] == "function"){
6454                         // shared options
6455                         listen(element, e, o, o[e], o.scope);
6456                     }else{
6457                         // individual options
6458                         listen(element, e, o[e]);
6459                     }
6460                 }
6461                 return;
6462             }
6463             return listen(element, eventName, options, fn, scope);
6464         },
6465         
6466         /**
6467          * Removes an event handler
6468          *
6469          * @param {String/HTMLElement}   element        The id or html element to remove the 
6470          *                             event from
6471          * @param {String}   eventName     The type of event
6472          * @param {Function} fn
6473          * @return {Boolean} True if a listener was actually removed
6474          */
6475         removeListener : function(element, eventName, fn){
6476             return stopListening(element, eventName, fn);
6477         },
6478         
6479         /**
6480          * Fires when the document is ready (before onload and before images are loaded). Can be 
6481          * accessed shorthanded Roo.onReady().
6482          * @param {Function} fn        The method the event invokes
6483          * @param {Object}   scope    An  object that becomes the scope of the handler
6484          * @param {boolean}  options
6485          */
6486         onDocumentReady : function(fn, scope, options){
6487             if(docReadyState){ // if it already fired
6488                 docReadyEvent.addListener(fn, scope, options);
6489                 docReadyEvent.fire();
6490                 docReadyEvent.clearListeners();
6491                 return;
6492             }
6493             if(!docReadyEvent){
6494                 initDocReady();
6495             }
6496             docReadyEvent.addListener(fn, scope, options);
6497         },
6498         
6499         /**
6500          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6501          * @param {Function} fn        The method the event invokes
6502          * @param {Object}   scope    An object that becomes the scope of the handler
6503          * @param {boolean}  options
6504          */
6505         onWindowResize : function(fn, scope, options){
6506             if(!resizeEvent){
6507                 resizeEvent = new Roo.util.Event();
6508                 resizeTask = new Roo.util.DelayedTask(function(){
6509                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6510                 });
6511                 E.on(window, "resize", function(){
6512                     if(Roo.isIE){
6513                         resizeTask.delay(50);
6514                     }else{
6515                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6516                     }
6517                 });
6518             }
6519             resizeEvent.addListener(fn, scope, options);
6520         },
6521
6522         /**
6523          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6524          * @param {Function} fn        The method the event invokes
6525          * @param {Object}   scope    An object that becomes the scope of the handler
6526          * @param {boolean}  options
6527          */
6528         onTextResize : function(fn, scope, options){
6529             if(!textEvent){
6530                 textEvent = new Roo.util.Event();
6531                 var textEl = new Roo.Element(document.createElement('div'));
6532                 textEl.dom.className = 'x-text-resize';
6533                 textEl.dom.innerHTML = 'X';
6534                 textEl.appendTo(document.body);
6535                 textSize = textEl.dom.offsetHeight;
6536                 setInterval(function(){
6537                     if(textEl.dom.offsetHeight != textSize){
6538                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6539                     }
6540                 }, this.textResizeInterval);
6541             }
6542             textEvent.addListener(fn, scope, options);
6543         },
6544
6545         /**
6546          * Removes the passed window resize listener.
6547          * @param {Function} fn        The method the event invokes
6548          * @param {Object}   scope    The scope of handler
6549          */
6550         removeResizeListener : function(fn, scope){
6551             if(resizeEvent){
6552                 resizeEvent.removeListener(fn, scope);
6553             }
6554         },
6555
6556         // private
6557         fireResize : function(){
6558             if(resizeEvent){
6559                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6560             }   
6561         },
6562         /**
6563          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6564          */
6565         ieDeferSrc : false,
6566         /**
6567          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6568          */
6569         textResizeInterval : 50
6570     };
6571     
6572     /**
6573      * Fix for doc tools
6574      * @scopeAlias pub=Roo.EventManager
6575      */
6576     
6577      /**
6578      * Appends an event handler to an element (shorthand for addListener)
6579      * @param {String/HTMLElement}   element        The html element or id to assign the
6580      * @param {String}   eventName The type of event to listen for
6581      * @param {Function} handler The method the event invokes
6582      * @param {Object}   scope (optional) The scope in which to execute the handler
6583      * function. The handler function's "this" context.
6584      * @param {Object}   options (optional) An object containing handler configuration
6585      * properties. This may contain any of the following properties:<ul>
6586      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6587      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6588      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6589      * <li>preventDefault {Boolean} True to prevent the default action</li>
6590      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6591      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6592      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6593      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6594      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6595      * by the specified number of milliseconds. If the event fires again within that time, the original
6596      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6597      * </ul><br>
6598      * <p>
6599      * <b>Combining Options</b><br>
6600      * Using the options argument, it is possible to combine different types of listeners:<br>
6601      * <br>
6602      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6603      * Code:<pre><code>
6604 el.on('click', this.onClick, this, {
6605     single: true,
6606     delay: 100,
6607     stopEvent : true,
6608     forumId: 4
6609 });</code></pre>
6610      * <p>
6611      * <b>Attaching multiple handlers in 1 call</b><br>
6612       * The method also allows for a single argument to be passed which is a config object containing properties
6613      * which specify multiple handlers.
6614      * <p>
6615      * Code:<pre><code>
6616 el.on({
6617     'click' : {
6618         fn: this.onClick
6619         scope: this,
6620         delay: 100
6621     },
6622     'mouseover' : {
6623         fn: this.onMouseOver
6624         scope: this
6625     },
6626     'mouseout' : {
6627         fn: this.onMouseOut
6628         scope: this
6629     }
6630 });</code></pre>
6631      * <p>
6632      * Or a shorthand syntax:<br>
6633      * Code:<pre><code>
6634 el.on({
6635     'click' : this.onClick,
6636     'mouseover' : this.onMouseOver,
6637     'mouseout' : this.onMouseOut
6638     scope: this
6639 });</code></pre>
6640      */
6641     pub.on = pub.addListener;
6642     pub.un = pub.removeListener;
6643
6644     pub.stoppedMouseDownEvent = new Roo.util.Event();
6645     return pub;
6646 }();
6647 /**
6648   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6649   * @param {Function} fn        The method the event invokes
6650   * @param {Object}   scope    An  object that becomes the scope of the handler
6651   * @param {boolean}  override If true, the obj passed in becomes
6652   *                             the execution scope of the listener
6653   * @member Roo
6654   * @method onReady
6655  */
6656 Roo.onReady = Roo.EventManager.onDocumentReady;
6657
6658 Roo.onReady(function(){
6659     var bd = Roo.get(document.body);
6660     if(!bd){ return; }
6661
6662     var cls = [
6663             Roo.isIE ? "roo-ie"
6664             : Roo.isIE11 ? "roo-ie11"
6665             : Roo.isEdge ? "roo-edge"
6666             : Roo.isGecko ? "roo-gecko"
6667             : Roo.isOpera ? "roo-opera"
6668             : Roo.isSafari ? "roo-safari" : ""];
6669
6670     if(Roo.isMac){
6671         cls.push("roo-mac");
6672     }
6673     if(Roo.isLinux){
6674         cls.push("roo-linux");
6675     }
6676     if(Roo.isIOS){
6677         cls.push("roo-ios");
6678     }
6679     if(Roo.isTouch){
6680         cls.push("roo-touch");
6681     }
6682     if(Roo.isBorderBox){
6683         cls.push('roo-border-box');
6684     }
6685     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6686         var p = bd.dom.parentNode;
6687         if(p){
6688             p.className += ' roo-strict';
6689         }
6690     }
6691     bd.addClass(cls.join(' '));
6692 });
6693
6694 /**
6695  * @class Roo.EventObject
6696  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6697  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6698  * Example:
6699  * <pre><code>
6700  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6701     e.preventDefault();
6702     var target = e.getTarget();
6703     ...
6704  }
6705  var myDiv = Roo.get("myDiv");
6706  myDiv.on("click", handleClick);
6707  //or
6708  Roo.EventManager.on("myDiv", 'click', handleClick);
6709  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6710  </code></pre>
6711  * @singleton
6712  */
6713 Roo.EventObject = function(){
6714     
6715     var E = Roo.lib.Event;
6716     
6717     // safari keypress events for special keys return bad keycodes
6718     var safariKeys = {
6719         63234 : 37, // left
6720         63235 : 39, // right
6721         63232 : 38, // up
6722         63233 : 40, // down
6723         63276 : 33, // page up
6724         63277 : 34, // page down
6725         63272 : 46, // delete
6726         63273 : 36, // home
6727         63275 : 35  // end
6728     };
6729
6730     // normalize button clicks
6731     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6732                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6733
6734     Roo.EventObjectImpl = function(e){
6735         if(e){
6736             this.setEvent(e.browserEvent || e);
6737         }
6738     };
6739     Roo.EventObjectImpl.prototype = {
6740         /**
6741          * Used to fix doc tools.
6742          * @scope Roo.EventObject.prototype
6743          */
6744             
6745
6746         
6747         
6748         /** The normal browser event */
6749         browserEvent : null,
6750         /** The button pressed in a mouse event */
6751         button : -1,
6752         /** True if the shift key was down during the event */
6753         shiftKey : false,
6754         /** True if the control key was down during the event */
6755         ctrlKey : false,
6756         /** True if the alt key was down during the event */
6757         altKey : false,
6758
6759         /** Key constant 
6760         * @type Number */
6761         BACKSPACE : 8,
6762         /** Key constant 
6763         * @type Number */
6764         TAB : 9,
6765         /** Key constant 
6766         * @type Number */
6767         RETURN : 13,
6768         /** Key constant 
6769         * @type Number */
6770         ENTER : 13,
6771         /** Key constant 
6772         * @type Number */
6773         SHIFT : 16,
6774         /** Key constant 
6775         * @type Number */
6776         CONTROL : 17,
6777         /** Key constant 
6778         * @type Number */
6779         ESC : 27,
6780         /** Key constant 
6781         * @type Number */
6782         SPACE : 32,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEUP : 33,
6786         /** Key constant 
6787         * @type Number */
6788         PAGEDOWN : 34,
6789         /** Key constant 
6790         * @type Number */
6791         END : 35,
6792         /** Key constant 
6793         * @type Number */
6794         HOME : 36,
6795         /** Key constant 
6796         * @type Number */
6797         LEFT : 37,
6798         /** Key constant 
6799         * @type Number */
6800         UP : 38,
6801         /** Key constant 
6802         * @type Number */
6803         RIGHT : 39,
6804         /** Key constant 
6805         * @type Number */
6806         DOWN : 40,
6807         /** Key constant 
6808         * @type Number */
6809         DELETE : 46,
6810         /** Key constant 
6811         * @type Number */
6812         F5 : 116,
6813
6814            /** @private */
6815         setEvent : function(e){
6816             if(e == this || (e && e.browserEvent)){ // already wrapped
6817                 return e;
6818             }
6819             this.browserEvent = e;
6820             if(e){
6821                 // normalize buttons
6822                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6823                 if(e.type == 'click' && this.button == -1){
6824                     this.button = 0;
6825                 }
6826                 this.type = e.type;
6827                 this.shiftKey = e.shiftKey;
6828                 // mac metaKey behaves like ctrlKey
6829                 this.ctrlKey = e.ctrlKey || e.metaKey;
6830                 this.altKey = e.altKey;
6831                 // in getKey these will be normalized for the mac
6832                 this.keyCode = e.keyCode;
6833                 // keyup warnings on firefox.
6834                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6835                 // cache the target for the delayed and or buffered events
6836                 this.target = E.getTarget(e);
6837                 // same for XY
6838                 this.xy = E.getXY(e);
6839             }else{
6840                 this.button = -1;
6841                 this.shiftKey = false;
6842                 this.ctrlKey = false;
6843                 this.altKey = false;
6844                 this.keyCode = 0;
6845                 this.charCode =0;
6846                 this.target = null;
6847                 this.xy = [0, 0];
6848             }
6849             return this;
6850         },
6851
6852         /**
6853          * Stop the event (preventDefault and stopPropagation)
6854          */
6855         stopEvent : function(){
6856             if(this.browserEvent){
6857                 if(this.browserEvent.type == 'mousedown'){
6858                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6859                 }
6860                 E.stopEvent(this.browserEvent);
6861             }
6862         },
6863
6864         /**
6865          * Prevents the browsers default handling of the event.
6866          */
6867         preventDefault : function(){
6868             if(this.browserEvent){
6869                 E.preventDefault(this.browserEvent);
6870             }
6871         },
6872
6873         /** @private */
6874         isNavKeyPress : function(){
6875             var k = this.keyCode;
6876             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6877             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6878         },
6879
6880         isSpecialKey : function(){
6881             var k = this.keyCode;
6882             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6883             (k == 16) || (k == 17) ||
6884             (k >= 18 && k <= 20) ||
6885             (k >= 33 && k <= 35) ||
6886             (k >= 36 && k <= 39) ||
6887             (k >= 44 && k <= 45);
6888         },
6889         /**
6890          * Cancels bubbling of the event.
6891          */
6892         stopPropagation : function(){
6893             if(this.browserEvent){
6894                 if(this.type == 'mousedown'){
6895                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6896                 }
6897                 E.stopPropagation(this.browserEvent);
6898             }
6899         },
6900
6901         /**
6902          * Gets the key code for the event.
6903          * @return {Number}
6904          */
6905         getCharCode : function(){
6906             return this.charCode || this.keyCode;
6907         },
6908
6909         /**
6910          * Returns a normalized keyCode for the event.
6911          * @return {Number} The key code
6912          */
6913         getKey : function(){
6914             var k = this.keyCode || this.charCode;
6915             return Roo.isSafari ? (safariKeys[k] || k) : k;
6916         },
6917
6918         /**
6919          * Gets the x coordinate of the event.
6920          * @return {Number}
6921          */
6922         getPageX : function(){
6923             return this.xy[0];
6924         },
6925
6926         /**
6927          * Gets the y coordinate of the event.
6928          * @return {Number}
6929          */
6930         getPageY : function(){
6931             return this.xy[1];
6932         },
6933
6934         /**
6935          * Gets the time of the event.
6936          * @return {Number}
6937          */
6938         getTime : function(){
6939             if(this.browserEvent){
6940                 return E.getTime(this.browserEvent);
6941             }
6942             return null;
6943         },
6944
6945         /**
6946          * Gets the page coordinates of the event.
6947          * @return {Array} The xy values like [x, y]
6948          */
6949         getXY : function(){
6950             return this.xy;
6951         },
6952
6953         /**
6954          * Gets the target for the event.
6955          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6956          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6957                 search as a number or element (defaults to 10 || document.body)
6958          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6959          * @return {HTMLelement}
6960          */
6961         getTarget : function(selector, maxDepth, returnEl){
6962             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6963         },
6964         /**
6965          * Gets the related target.
6966          * @return {HTMLElement}
6967          */
6968         getRelatedTarget : function(){
6969             if(this.browserEvent){
6970                 return E.getRelatedTarget(this.browserEvent);
6971             }
6972             return null;
6973         },
6974
6975         /**
6976          * Normalizes mouse wheel delta across browsers
6977          * @return {Number} The delta
6978          */
6979         getWheelDelta : function(){
6980             var e = this.browserEvent;
6981             var delta = 0;
6982             if(e.wheelDelta){ /* IE/Opera. */
6983                 delta = e.wheelDelta/120;
6984             }else if(e.detail){ /* Mozilla case. */
6985                 delta = -e.detail/3;
6986             }
6987             return delta;
6988         },
6989
6990         /**
6991          * Returns true if the control, meta, shift or alt key was pressed during this event.
6992          * @return {Boolean}
6993          */
6994         hasModifier : function(){
6995             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6996         },
6997
6998         /**
6999          * Returns true if the target of this event equals el or is a child of el
7000          * @param {String/HTMLElement/Element} el
7001          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
7002          * @return {Boolean}
7003          */
7004         within : function(el, related){
7005             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7006             return t && Roo.fly(el).contains(t);
7007         },
7008
7009         getPoint : function(){
7010             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7011         }
7012     };
7013
7014     return new Roo.EventObjectImpl();
7015 }();
7016             
7017     /*
7018  * Based on:
7019  * Ext JS Library 1.1.1
7020  * Copyright(c) 2006-2007, Ext JS, LLC.
7021  *
7022  * Originally Released Under LGPL - original licence link has changed is not relivant.
7023  *
7024  * Fork - LGPL
7025  * <script type="text/javascript">
7026  */
7027
7028  
7029 // was in Composite Element!??!?!
7030  
7031 (function(){
7032     var D = Roo.lib.Dom;
7033     var E = Roo.lib.Event;
7034     var A = Roo.lib.Anim;
7035
7036     // local style camelizing for speed
7037     var propCache = {};
7038     var camelRe = /(-[a-z])/gi;
7039     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7040     var view = document.defaultView;
7041
7042 /**
7043  * @class Roo.Element
7044  * Represents an Element in the DOM.<br><br>
7045  * Usage:<br>
7046 <pre><code>
7047 var el = Roo.get("my-div");
7048
7049 // or with getEl
7050 var el = getEl("my-div");
7051
7052 // or with a DOM element
7053 var el = Roo.get(myDivElement);
7054 </code></pre>
7055  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7056  * each call instead of constructing a new one.<br><br>
7057  * <b>Animations</b><br />
7058  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7059  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7060 <pre>
7061 Option    Default   Description
7062 --------- --------  ---------------------------------------------
7063 duration  .35       The duration of the animation in seconds
7064 easing    easeOut   The YUI easing method
7065 callback  none      A function to execute when the anim completes
7066 scope     this      The scope (this) of the callback function
7067 </pre>
7068 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7069 * manipulate the animation. Here's an example:
7070 <pre><code>
7071 var el = Roo.get("my-div");
7072
7073 // no animation
7074 el.setWidth(100);
7075
7076 // default animation
7077 el.setWidth(100, true);
7078
7079 // animation with some options set
7080 el.setWidth(100, {
7081     duration: 1,
7082     callback: this.foo,
7083     scope: this
7084 });
7085
7086 // using the "anim" property to get the Anim object
7087 var opt = {
7088     duration: 1,
7089     callback: this.foo,
7090     scope: this
7091 };
7092 el.setWidth(100, opt);
7093 ...
7094 if(opt.anim.isAnimated()){
7095     opt.anim.stop();
7096 }
7097 </code></pre>
7098 * <b> Composite (Collections of) Elements</b><br />
7099  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7100  * @constructor Create a new Element directly.
7101  * @param {String/HTMLElement} element
7102  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7103  */
7104     Roo.Element = function(element, forceNew){
7105         var dom = typeof element == "string" ?
7106                 document.getElementById(element) : element;
7107         if(!dom){ // invalid id/element
7108             return null;
7109         }
7110         var id = dom.id;
7111         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7112             return Roo.Element.cache[id];
7113         }
7114
7115         /**
7116          * The DOM element
7117          * @type HTMLElement
7118          */
7119         this.dom = dom;
7120
7121         /**
7122          * The DOM element ID
7123          * @type String
7124          */
7125         this.id = id || Roo.id(dom);
7126     };
7127
7128     var El = Roo.Element;
7129
7130     El.prototype = {
7131         /**
7132          * The element's default display mode  (defaults to "")
7133          * @type String
7134          */
7135         originalDisplay : "",
7136
7137         visibilityMode : 1,
7138         /**
7139          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7140          * @type String
7141          */
7142         defaultUnit : "px",
7143         
7144         /**
7145          * Sets the element's visibility mode. When setVisible() is called it
7146          * will use this to determine whether to set the visibility or the display property.
7147          * @param visMode Element.VISIBILITY or Element.DISPLAY
7148          * @return {Roo.Element} this
7149          */
7150         setVisibilityMode : function(visMode){
7151             this.visibilityMode = visMode;
7152             return this;
7153         },
7154         /**
7155          * Convenience method for setVisibilityMode(Element.DISPLAY)
7156          * @param {String} display (optional) What to set display to when visible
7157          * @return {Roo.Element} this
7158          */
7159         enableDisplayMode : function(display){
7160             this.setVisibilityMode(El.DISPLAY);
7161             if(typeof display != "undefined") { this.originalDisplay = display; }
7162             return this;
7163         },
7164
7165         /**
7166          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7167          * @param {String} selector The simple selector to test
7168          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7169                 search as a number or element (defaults to 10 || document.body)
7170          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7171          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7172          */
7173         findParent : function(simpleSelector, maxDepth, returnEl){
7174             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7175             maxDepth = maxDepth || 50;
7176             if(typeof maxDepth != "number"){
7177                 stopEl = Roo.getDom(maxDepth);
7178                 maxDepth = 10;
7179             }
7180             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7181                 if(dq.is(p, simpleSelector)){
7182                     return returnEl ? Roo.get(p) : p;
7183                 }
7184                 depth++;
7185                 p = p.parentNode;
7186             }
7187             return null;
7188         },
7189
7190
7191         /**
7192          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7193          * @param {String} selector The simple selector to test
7194          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7195                 search as a number or element (defaults to 10 || document.body)
7196          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7197          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7198          */
7199         findParentNode : function(simpleSelector, maxDepth, returnEl){
7200             var p = Roo.fly(this.dom.parentNode, '_internal');
7201             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7202         },
7203         
7204         /**
7205          * Looks at  the scrollable parent element
7206          */
7207         findScrollableParent : function()
7208         {
7209             var overflowRegex = /(auto|scroll)/;
7210             
7211             if(this.getStyle('position') === 'fixed'){
7212                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7213             }
7214             
7215             var excludeStaticParent = this.getStyle('position') === "absolute";
7216             
7217             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7218                 
7219                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7220                     continue;
7221                 }
7222                 
7223                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7224                     return parent;
7225                 }
7226                 
7227                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7228                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7229                 }
7230             }
7231             
7232             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7233         },
7234
7235         /**
7236          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7237          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7238          * @param {String} selector The simple selector to test
7239          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7240                 search as a number or element (defaults to 10 || document.body)
7241          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7242          */
7243         up : function(simpleSelector, maxDepth){
7244             return this.findParentNode(simpleSelector, maxDepth, true);
7245         },
7246
7247
7248
7249         /**
7250          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7251          * @param {String} selector The simple selector to test
7252          * @return {Boolean} True if this element matches the selector, else false
7253          */
7254         is : function(simpleSelector){
7255             return Roo.DomQuery.is(this.dom, simpleSelector);
7256         },
7257
7258         /**
7259          * Perform animation on this element.
7260          * @param {Object} args The YUI animation control args
7261          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7262          * @param {Function} onComplete (optional) Function to call when animation completes
7263          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7264          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7265          * @return {Roo.Element} this
7266          */
7267         animate : function(args, duration, onComplete, easing, animType){
7268             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7269             return this;
7270         },
7271
7272         /*
7273          * @private Internal animation call
7274          */
7275         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7276             animType = animType || 'run';
7277             opt = opt || {};
7278             var anim = Roo.lib.Anim[animType](
7279                 this.dom, args,
7280                 (opt.duration || defaultDur) || .35,
7281                 (opt.easing || defaultEase) || 'easeOut',
7282                 function(){
7283                     Roo.callback(cb, this);
7284                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7285                 },
7286                 this
7287             );
7288             opt.anim = anim;
7289             return anim;
7290         },
7291
7292         // private legacy anim prep
7293         preanim : function(a, i){
7294             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7295         },
7296
7297         /**
7298          * Removes worthless text nodes
7299          * @param {Boolean} forceReclean (optional) By default the element
7300          * keeps track if it has been cleaned already so
7301          * you can call this over and over. However, if you update the element and
7302          * need to force a reclean, you can pass true.
7303          */
7304         clean : function(forceReclean){
7305             if(this.isCleaned && forceReclean !== true){
7306                 return this;
7307             }
7308             var ns = /\S/;
7309             var d = this.dom, n = d.firstChild, ni = -1;
7310             while(n){
7311                 var nx = n.nextSibling;
7312                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7313                     d.removeChild(n);
7314                 }else{
7315                     n.nodeIndex = ++ni;
7316                 }
7317                 n = nx;
7318             }
7319             this.isCleaned = true;
7320             return this;
7321         },
7322
7323         // private
7324         calcOffsetsTo : function(el){
7325             el = Roo.get(el);
7326             var d = el.dom;
7327             var restorePos = false;
7328             if(el.getStyle('position') == 'static'){
7329                 el.position('relative');
7330                 restorePos = true;
7331             }
7332             var x = 0, y =0;
7333             var op = this.dom;
7334             while(op && op != d && op.tagName != 'HTML'){
7335                 x+= op.offsetLeft;
7336                 y+= op.offsetTop;
7337                 op = op.offsetParent;
7338             }
7339             if(restorePos){
7340                 el.position('static');
7341             }
7342             return [x, y];
7343         },
7344
7345         /**
7346          * Scrolls this element into view within the passed container.
7347          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7348          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7349          * @return {Roo.Element} this
7350          */
7351         scrollIntoView : function(container, hscroll){
7352             var c = Roo.getDom(container) || document.body;
7353             var el = this.dom;
7354
7355             var o = this.calcOffsetsTo(c),
7356                 l = o[0],
7357                 t = o[1],
7358                 b = t+el.offsetHeight,
7359                 r = l+el.offsetWidth;
7360
7361             var ch = c.clientHeight;
7362             var ct = parseInt(c.scrollTop, 10);
7363             var cl = parseInt(c.scrollLeft, 10);
7364             var cb = ct + ch;
7365             var cr = cl + c.clientWidth;
7366
7367             if(t < ct){
7368                 c.scrollTop = t;
7369             }else if(b > cb){
7370                 c.scrollTop = b-ch;
7371             }
7372
7373             if(hscroll !== false){
7374                 if(l < cl){
7375                     c.scrollLeft = l;
7376                 }else if(r > cr){
7377                     c.scrollLeft = r-c.clientWidth;
7378                 }
7379             }
7380             return this;
7381         },
7382
7383         // private
7384         scrollChildIntoView : function(child, hscroll){
7385             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7386         },
7387
7388         /**
7389          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7390          * the new height may not be available immediately.
7391          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7392          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7393          * @param {Function} onComplete (optional) Function to call when animation completes
7394          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7395          * @return {Roo.Element} this
7396          */
7397         autoHeight : function(animate, duration, onComplete, easing){
7398             var oldHeight = this.getHeight();
7399             this.clip();
7400             this.setHeight(1); // force clipping
7401             setTimeout(function(){
7402                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7403                 if(!animate){
7404                     this.setHeight(height);
7405                     this.unclip();
7406                     if(typeof onComplete == "function"){
7407                         onComplete();
7408                     }
7409                 }else{
7410                     this.setHeight(oldHeight); // restore original height
7411                     this.setHeight(height, animate, duration, function(){
7412                         this.unclip();
7413                         if(typeof onComplete == "function") { onComplete(); }
7414                     }.createDelegate(this), easing);
7415                 }
7416             }.createDelegate(this), 0);
7417             return this;
7418         },
7419
7420         /**
7421          * Returns true if this element is an ancestor of the passed element
7422          * @param {HTMLElement/String} el The element to check
7423          * @return {Boolean} True if this element is an ancestor of el, else false
7424          */
7425         contains : function(el){
7426             if(!el){return false;}
7427             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7428         },
7429
7430         /**
7431          * Checks whether the element is currently visible using both visibility and display properties.
7432          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7433          * @return {Boolean} True if the element is currently visible, else false
7434          */
7435         isVisible : function(deep) {
7436             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7437             if(deep !== true || !vis){
7438                 return vis;
7439             }
7440             var p = this.dom.parentNode;
7441             while(p && p.tagName.toLowerCase() != "body"){
7442                 if(!Roo.fly(p, '_isVisible').isVisible()){
7443                     return false;
7444                 }
7445                 p = p.parentNode;
7446             }
7447             return true;
7448         },
7449
7450         /**
7451          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7452          * @param {String} selector The CSS selector
7453          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7454          * @return {CompositeElement/CompositeElementLite} The composite element
7455          */
7456         select : function(selector, unique){
7457             return El.select(selector, unique, this.dom);
7458         },
7459
7460         /**
7461          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7462          * @param {String} selector The CSS selector
7463          * @return {Array} An array of the matched nodes
7464          */
7465         query : function(selector, unique){
7466             return Roo.DomQuery.select(selector, this.dom);
7467         },
7468
7469         /**
7470          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7471          * @param {String} selector The CSS selector
7472          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7473          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7474          */
7475         child : function(selector, returnDom){
7476             var n = Roo.DomQuery.selectNode(selector, this.dom);
7477             return returnDom ? n : Roo.get(n);
7478         },
7479
7480         /**
7481          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7482          * @param {String} selector The CSS selector
7483          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7484          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7485          */
7486         down : function(selector, returnDom){
7487             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7488             return returnDom ? n : Roo.get(n);
7489         },
7490
7491         /**
7492          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7493          * @param {String} group The group the DD object is member of
7494          * @param {Object} config The DD config object
7495          * @param {Object} overrides An object containing methods to override/implement on the DD object
7496          * @return {Roo.dd.DD} The DD object
7497          */
7498         initDD : function(group, config, overrides){
7499             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7500             return Roo.apply(dd, overrides);
7501         },
7502
7503         /**
7504          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7505          * @param {String} group The group the DDProxy object is member of
7506          * @param {Object} config The DDProxy config object
7507          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7508          * @return {Roo.dd.DDProxy} The DDProxy object
7509          */
7510         initDDProxy : function(group, config, overrides){
7511             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7512             return Roo.apply(dd, overrides);
7513         },
7514
7515         /**
7516          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7517          * @param {String} group The group the DDTarget object is member of
7518          * @param {Object} config The DDTarget config object
7519          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7520          * @return {Roo.dd.DDTarget} The DDTarget object
7521          */
7522         initDDTarget : function(group, config, overrides){
7523             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7524             return Roo.apply(dd, overrides);
7525         },
7526
7527         /**
7528          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7529          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7530          * @param {Boolean} visible Whether the element is visible
7531          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7532          * @return {Roo.Element} this
7533          */
7534          setVisible : function(visible, animate){
7535             if(!animate || !A){
7536                 if(this.visibilityMode == El.DISPLAY){
7537                     this.setDisplayed(visible);
7538                 }else{
7539                     this.fixDisplay();
7540                     this.dom.style.visibility = visible ? "visible" : "hidden";
7541                 }
7542             }else{
7543                 // closure for composites
7544                 var dom = this.dom;
7545                 var visMode = this.visibilityMode;
7546                 if(visible){
7547                     this.setOpacity(.01);
7548                     this.setVisible(true);
7549                 }
7550                 this.anim({opacity: { to: (visible?1:0) }},
7551                       this.preanim(arguments, 1),
7552                       null, .35, 'easeIn', function(){
7553                          if(!visible){
7554                              if(visMode == El.DISPLAY){
7555                                  dom.style.display = "none";
7556                              }else{
7557                                  dom.style.visibility = "hidden";
7558                              }
7559                              Roo.get(dom).setOpacity(1);
7560                          }
7561                      });
7562             }
7563             return this;
7564         },
7565
7566         /**
7567          * Returns true if display is not "none"
7568          * @return {Boolean}
7569          */
7570         isDisplayed : function() {
7571             return this.getStyle("display") != "none";
7572         },
7573
7574         /**
7575          * Toggles the element's visibility or display, depending on visibility mode.
7576          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7577          * @return {Roo.Element} this
7578          */
7579         toggle : function(animate){
7580             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7581             return this;
7582         },
7583
7584         /**
7585          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7586          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7587          * @return {Roo.Element} this
7588          */
7589         setDisplayed : function(value) {
7590             if(typeof value == "boolean"){
7591                value = value ? this.originalDisplay : "none";
7592             }
7593             this.setStyle("display", value);
7594             return this;
7595         },
7596
7597         /**
7598          * Tries to focus the element. Any exceptions are caught and ignored.
7599          * @return {Roo.Element} this
7600          */
7601         focus : function() {
7602             try{
7603                 this.dom.focus();
7604             }catch(e){}
7605             return this;
7606         },
7607
7608         /**
7609          * Tries to blur the element. Any exceptions are caught and ignored.
7610          * @return {Roo.Element} this
7611          */
7612         blur : function() {
7613             try{
7614                 this.dom.blur();
7615             }catch(e){}
7616             return this;
7617         },
7618
7619         /**
7620          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7621          * @param {String/Array} className The CSS class to add, or an array of classes
7622          * @return {Roo.Element} this
7623          */
7624         addClass : function(className){
7625             if(className instanceof Array){
7626                 for(var i = 0, len = className.length; i < len; i++) {
7627                     this.addClass(className[i]);
7628                 }
7629             }else{
7630                 if(className && !this.hasClass(className)){
7631                     this.dom.className = this.dom.className + " " + className;
7632                 }
7633             }
7634             return this;
7635         },
7636
7637         /**
7638          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7639          * @param {String/Array} className The CSS class to add, or an array of classes
7640          * @return {Roo.Element} this
7641          */
7642         radioClass : function(className){
7643             var siblings = this.dom.parentNode.childNodes;
7644             for(var i = 0; i < siblings.length; i++) {
7645                 var s = siblings[i];
7646                 if(s.nodeType == 1){
7647                     Roo.get(s).removeClass(className);
7648                 }
7649             }
7650             this.addClass(className);
7651             return this;
7652         },
7653
7654         /**
7655          * Removes one or more CSS classes from the element.
7656          * @param {String/Array} className The CSS class to remove, or an array of classes
7657          * @return {Roo.Element} this
7658          */
7659         removeClass : function(className){
7660             if(!className || !this.dom.className){
7661                 return this;
7662             }
7663             if(className instanceof Array){
7664                 for(var i = 0, len = className.length; i < len; i++) {
7665                     this.removeClass(className[i]);
7666                 }
7667             }else{
7668                 if(this.hasClass(className)){
7669                     var re = this.classReCache[className];
7670                     if (!re) {
7671                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7672                        this.classReCache[className] = re;
7673                     }
7674                     this.dom.className =
7675                         this.dom.className.replace(re, " ");
7676                 }
7677             }
7678             return this;
7679         },
7680
7681         // private
7682         classReCache: {},
7683
7684         /**
7685          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7686          * @param {String} className The CSS class to toggle
7687          * @return {Roo.Element} this
7688          */
7689         toggleClass : function(className){
7690             if(this.hasClass(className)){
7691                 this.removeClass(className);
7692             }else{
7693                 this.addClass(className);
7694             }
7695             return this;
7696         },
7697
7698         /**
7699          * Checks if the specified CSS class exists on this element's DOM node.
7700          * @param {String} className The CSS class to check for
7701          * @return {Boolean} True if the class exists, else false
7702          */
7703         hasClass : function(className){
7704             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7705         },
7706
7707         /**
7708          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7709          * @param {String} oldClassName The CSS class to replace
7710          * @param {String} newClassName The replacement CSS class
7711          * @return {Roo.Element} this
7712          */
7713         replaceClass : function(oldClassName, newClassName){
7714             this.removeClass(oldClassName);
7715             this.addClass(newClassName);
7716             return this;
7717         },
7718
7719         /**
7720          * Returns an object with properties matching the styles requested.
7721          * For example, el.getStyles('color', 'font-size', 'width') might return
7722          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7723          * @param {String} style1 A style name
7724          * @param {String} style2 A style name
7725          * @param {String} etc.
7726          * @return {Object} The style object
7727          */
7728         getStyles : function(){
7729             var a = arguments, len = a.length, r = {};
7730             for(var i = 0; i < len; i++){
7731                 r[a[i]] = this.getStyle(a[i]);
7732             }
7733             return r;
7734         },
7735
7736         /**
7737          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7738          * @param {String} property The style property whose value is returned.
7739          * @return {String} The current value of the style property for this element.
7740          */
7741         getStyle : function(){
7742             return view && view.getComputedStyle ?
7743                 function(prop){
7744                     var el = this.dom, v, cs, camel;
7745                     if(prop == 'float'){
7746                         prop = "cssFloat";
7747                     }
7748                     if(el.style && (v = el.style[prop])){
7749                         return v;
7750                     }
7751                     if(cs = view.getComputedStyle(el, "")){
7752                         if(!(camel = propCache[prop])){
7753                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7754                         }
7755                         return cs[camel];
7756                     }
7757                     return null;
7758                 } :
7759                 function(prop){
7760                     var el = this.dom, v, cs, camel;
7761                     if(prop == 'opacity'){
7762                         if(typeof el.style.filter == 'string'){
7763                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7764                             if(m){
7765                                 var fv = parseFloat(m[1]);
7766                                 if(!isNaN(fv)){
7767                                     return fv ? fv / 100 : 0;
7768                                 }
7769                             }
7770                         }
7771                         return 1;
7772                     }else if(prop == 'float'){
7773                         prop = "styleFloat";
7774                     }
7775                     if(!(camel = propCache[prop])){
7776                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7777                     }
7778                     if(v = el.style[camel]){
7779                         return v;
7780                     }
7781                     if(cs = el.currentStyle){
7782                         return cs[camel];
7783                     }
7784                     return null;
7785                 };
7786         }(),
7787
7788         /**
7789          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7790          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7791          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7792          * @return {Roo.Element} this
7793          */
7794         setStyle : function(prop, value){
7795             if(typeof prop == "string"){
7796                 
7797                 if (prop == 'float') {
7798                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7799                     return this;
7800                 }
7801                 
7802                 var camel;
7803                 if(!(camel = propCache[prop])){
7804                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7805                 }
7806                 
7807                 if(camel == 'opacity') {
7808                     this.setOpacity(value);
7809                 }else{
7810                     this.dom.style[camel] = value;
7811                 }
7812             }else{
7813                 for(var style in prop){
7814                     if(typeof prop[style] != "function"){
7815                        this.setStyle(style, prop[style]);
7816                     }
7817                 }
7818             }
7819             return this;
7820         },
7821
7822         /**
7823          * More flexible version of {@link #setStyle} for setting style properties.
7824          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7825          * a function which returns such a specification.
7826          * @return {Roo.Element} this
7827          */
7828         applyStyles : function(style){
7829             Roo.DomHelper.applyStyles(this.dom, style);
7830             return this;
7831         },
7832
7833         /**
7834           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7835           * @return {Number} The X position of the element
7836           */
7837         getX : function(){
7838             return D.getX(this.dom);
7839         },
7840
7841         /**
7842           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7843           * @return {Number} The Y position of the element
7844           */
7845         getY : function(){
7846             return D.getY(this.dom);
7847         },
7848
7849         /**
7850           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7851           * @return {Array} The XY position of the element
7852           */
7853         getXY : function(){
7854             return D.getXY(this.dom);
7855         },
7856
7857         /**
7858          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7859          * @param {Number} The X position of the element
7860          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7861          * @return {Roo.Element} this
7862          */
7863         setX : function(x, animate){
7864             if(!animate || !A){
7865                 D.setX(this.dom, x);
7866             }else{
7867                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7868             }
7869             return this;
7870         },
7871
7872         /**
7873          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7874          * @param {Number} The Y position of the element
7875          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7876          * @return {Roo.Element} this
7877          */
7878         setY : function(y, animate){
7879             if(!animate || !A){
7880                 D.setY(this.dom, y);
7881             }else{
7882                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7883             }
7884             return this;
7885         },
7886
7887         /**
7888          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7889          * @param {String} left The left CSS property value
7890          * @return {Roo.Element} this
7891          */
7892         setLeft : function(left){
7893             this.setStyle("left", this.addUnits(left));
7894             return this;
7895         },
7896
7897         /**
7898          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7899          * @param {String} top The top CSS property value
7900          * @return {Roo.Element} this
7901          */
7902         setTop : function(top){
7903             this.setStyle("top", this.addUnits(top));
7904             return this;
7905         },
7906
7907         /**
7908          * Sets the element's CSS right style.
7909          * @param {String} right The right CSS property value
7910          * @return {Roo.Element} this
7911          */
7912         setRight : function(right){
7913             this.setStyle("right", this.addUnits(right));
7914             return this;
7915         },
7916
7917         /**
7918          * Sets the element's CSS bottom style.
7919          * @param {String} bottom The bottom CSS property value
7920          * @return {Roo.Element} this
7921          */
7922         setBottom : function(bottom){
7923             this.setStyle("bottom", this.addUnits(bottom));
7924             return this;
7925         },
7926
7927         /**
7928          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7929          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7930          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7931          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7932          * @return {Roo.Element} this
7933          */
7934         setXY : function(pos, animate){
7935             if(!animate || !A){
7936                 D.setXY(this.dom, pos);
7937             }else{
7938                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7939             }
7940             return this;
7941         },
7942
7943         /**
7944          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7945          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7946          * @param {Number} x X value for new position (coordinates are page-based)
7947          * @param {Number} y Y value for new position (coordinates are page-based)
7948          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7949          * @return {Roo.Element} this
7950          */
7951         setLocation : function(x, y, animate){
7952             this.setXY([x, y], this.preanim(arguments, 2));
7953             return this;
7954         },
7955
7956         /**
7957          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7958          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7959          * @param {Number} x X value for new position (coordinates are page-based)
7960          * @param {Number} y Y value for new position (coordinates are page-based)
7961          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7962          * @return {Roo.Element} this
7963          */
7964         moveTo : function(x, y, animate){
7965             this.setXY([x, y], this.preanim(arguments, 2));
7966             return this;
7967         },
7968
7969         /**
7970          * Returns the region of the given element.
7971          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7972          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7973          */
7974         getRegion : function(){
7975             return D.getRegion(this.dom);
7976         },
7977
7978         /**
7979          * Returns the offset height of the element
7980          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7981          * @return {Number} The element's height
7982          */
7983         getHeight : function(contentHeight){
7984             var h = this.dom.offsetHeight || 0;
7985             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7986         },
7987
7988         /**
7989          * Returns the offset width of the element
7990          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7991          * @return {Number} The element's width
7992          */
7993         getWidth : function(contentWidth){
7994             var w = this.dom.offsetWidth || 0;
7995             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7996         },
7997
7998         /**
7999          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
8000          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
8001          * if a height has not been set using CSS.
8002          * @return {Number}
8003          */
8004         getComputedHeight : function(){
8005             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8006             if(!h){
8007                 h = parseInt(this.getStyle('height'), 10) || 0;
8008                 if(!this.isBorderBox()){
8009                     h += this.getFrameWidth('tb');
8010                 }
8011             }
8012             return h;
8013         },
8014
8015         /**
8016          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8017          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8018          * if a width has not been set using CSS.
8019          * @return {Number}
8020          */
8021         getComputedWidth : function(){
8022             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8023             if(!w){
8024                 w = parseInt(this.getStyle('width'), 10) || 0;
8025                 if(!this.isBorderBox()){
8026                     w += this.getFrameWidth('lr');
8027                 }
8028             }
8029             return w;
8030         },
8031
8032         /**
8033          * Returns the size of the element.
8034          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8035          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8036          */
8037         getSize : function(contentSize){
8038             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8039         },
8040
8041         /**
8042          * Returns the width and height of the viewport.
8043          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8044          */
8045         getViewSize : function(){
8046             var d = this.dom, doc = document, aw = 0, ah = 0;
8047             if(d == doc || d == doc.body){
8048                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8049             }else{
8050                 return {
8051                     width : d.clientWidth,
8052                     height: d.clientHeight
8053                 };
8054             }
8055         },
8056
8057         /**
8058          * Returns the value of the "value" attribute
8059          * @param {Boolean} asNumber true to parse the value as a number
8060          * @return {String/Number}
8061          */
8062         getValue : function(asNumber){
8063             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8064         },
8065
8066         // private
8067         adjustWidth : function(width){
8068             if(typeof width == "number"){
8069                 if(this.autoBoxAdjust && !this.isBorderBox()){
8070                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8071                 }
8072                 if(width < 0){
8073                     width = 0;
8074                 }
8075             }
8076             return width;
8077         },
8078
8079         // private
8080         adjustHeight : function(height){
8081             if(typeof height == "number"){
8082                if(this.autoBoxAdjust && !this.isBorderBox()){
8083                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8084                }
8085                if(height < 0){
8086                    height = 0;
8087                }
8088             }
8089             return height;
8090         },
8091
8092         /**
8093          * Set the width of the element
8094          * @param {Number} width The new width
8095          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8096          * @return {Roo.Element} this
8097          */
8098         setWidth : function(width, animate){
8099             width = this.adjustWidth(width);
8100             if(!animate || !A){
8101                 this.dom.style.width = this.addUnits(width);
8102             }else{
8103                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8104             }
8105             return this;
8106         },
8107
8108         /**
8109          * Set the height of the element
8110          * @param {Number} height The new height
8111          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8112          * @return {Roo.Element} this
8113          */
8114          setHeight : function(height, animate){
8115             height = this.adjustHeight(height);
8116             if(!animate || !A){
8117                 this.dom.style.height = this.addUnits(height);
8118             }else{
8119                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8120             }
8121             return this;
8122         },
8123
8124         /**
8125          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8126          * @param {Number} width The new width
8127          * @param {Number} height The new height
8128          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8129          * @return {Roo.Element} this
8130          */
8131          setSize : function(width, height, animate){
8132             if(typeof width == "object"){ // in case of object from getSize()
8133                 height = width.height; width = width.width;
8134             }
8135             width = this.adjustWidth(width); height = this.adjustHeight(height);
8136             if(!animate || !A){
8137                 this.dom.style.width = this.addUnits(width);
8138                 this.dom.style.height = this.addUnits(height);
8139             }else{
8140                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8141             }
8142             return this;
8143         },
8144
8145         /**
8146          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8147          * @param {Number} x X value for new position (coordinates are page-based)
8148          * @param {Number} y Y value for new position (coordinates are page-based)
8149          * @param {Number} width The new width
8150          * @param {Number} height The new height
8151          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8152          * @return {Roo.Element} this
8153          */
8154         setBounds : function(x, y, width, height, animate){
8155             if(!animate || !A){
8156                 this.setSize(width, height);
8157                 this.setLocation(x, y);
8158             }else{
8159                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8160                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8161                               this.preanim(arguments, 4), 'motion');
8162             }
8163             return this;
8164         },
8165
8166         /**
8167          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8168          * @param {Roo.lib.Region} region The region to fill
8169          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8170          * @return {Roo.Element} this
8171          */
8172         setRegion : function(region, animate){
8173             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8174             return this;
8175         },
8176
8177         /**
8178          * Appends an event handler
8179          *
8180          * @param {String}   eventName     The type of event to append
8181          * @param {Function} fn        The method the event invokes
8182          * @param {Object} scope       (optional) The scope (this object) of the fn
8183          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8184          */
8185         addListener : function(eventName, fn, scope, options){
8186             if (this.dom) {
8187                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8188             }
8189         },
8190
8191         /**
8192          * Removes an event handler from this element
8193          * @param {String} eventName the type of event to remove
8194          * @param {Function} fn the method the event invokes
8195          * @return {Roo.Element} this
8196          */
8197         removeListener : function(eventName, fn){
8198             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8199             return this;
8200         },
8201
8202         /**
8203          * Removes all previous added listeners from this element
8204          * @return {Roo.Element} this
8205          */
8206         removeAllListeners : function(){
8207             E.purgeElement(this.dom);
8208             return this;
8209         },
8210
8211         relayEvent : function(eventName, observable){
8212             this.on(eventName, function(e){
8213                 observable.fireEvent(eventName, e);
8214             });
8215         },
8216
8217         /**
8218          * Set the opacity of the element
8219          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8220          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8221          * @return {Roo.Element} this
8222          */
8223          setOpacity : function(opacity, animate){
8224             if(!animate || !A){
8225                 var s = this.dom.style;
8226                 if(Roo.isIE){
8227                     s.zoom = 1;
8228                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8229                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8230                 }else{
8231                     s.opacity = opacity;
8232                 }
8233             }else{
8234                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8235             }
8236             return this;
8237         },
8238
8239         /**
8240          * Gets the left X coordinate
8241          * @param {Boolean} local True to get the local css position instead of page coordinate
8242          * @return {Number}
8243          */
8244         getLeft : function(local){
8245             if(!local){
8246                 return this.getX();
8247             }else{
8248                 return parseInt(this.getStyle("left"), 10) || 0;
8249             }
8250         },
8251
8252         /**
8253          * Gets the right X coordinate of the element (element X position + element width)
8254          * @param {Boolean} local True to get the local css position instead of page coordinate
8255          * @return {Number}
8256          */
8257         getRight : function(local){
8258             if(!local){
8259                 return this.getX() + this.getWidth();
8260             }else{
8261                 return (this.getLeft(true) + this.getWidth()) || 0;
8262             }
8263         },
8264
8265         /**
8266          * Gets the top Y coordinate
8267          * @param {Boolean} local True to get the local css position instead of page coordinate
8268          * @return {Number}
8269          */
8270         getTop : function(local) {
8271             if(!local){
8272                 return this.getY();
8273             }else{
8274                 return parseInt(this.getStyle("top"), 10) || 0;
8275             }
8276         },
8277
8278         /**
8279          * Gets the bottom Y coordinate of the element (element Y position + element height)
8280          * @param {Boolean} local True to get the local css position instead of page coordinate
8281          * @return {Number}
8282          */
8283         getBottom : function(local){
8284             if(!local){
8285                 return this.getY() + this.getHeight();
8286             }else{
8287                 return (this.getTop(true) + this.getHeight()) || 0;
8288             }
8289         },
8290
8291         /**
8292         * Initializes positioning on this element. If a desired position is not passed, it will make the
8293         * the element positioned relative IF it is not already positioned.
8294         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8295         * @param {Number} zIndex (optional) The zIndex to apply
8296         * @param {Number} x (optional) Set the page X position
8297         * @param {Number} y (optional) Set the page Y position
8298         */
8299         position : function(pos, zIndex, x, y){
8300             if(!pos){
8301                if(this.getStyle('position') == 'static'){
8302                    this.setStyle('position', 'relative');
8303                }
8304             }else{
8305                 this.setStyle("position", pos);
8306             }
8307             if(zIndex){
8308                 this.setStyle("z-index", zIndex);
8309             }
8310             if(x !== undefined && y !== undefined){
8311                 this.setXY([x, y]);
8312             }else if(x !== undefined){
8313                 this.setX(x);
8314             }else if(y !== undefined){
8315                 this.setY(y);
8316             }
8317         },
8318
8319         /**
8320         * Clear positioning back to the default when the document was loaded
8321         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8322         * @return {Roo.Element} this
8323          */
8324         clearPositioning : function(value){
8325             value = value ||'';
8326             this.setStyle({
8327                 "left": value,
8328                 "right": value,
8329                 "top": value,
8330                 "bottom": value,
8331                 "z-index": "",
8332                 "position" : "static"
8333             });
8334             return this;
8335         },
8336
8337         /**
8338         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8339         * snapshot before performing an update and then restoring the element.
8340         * @return {Object}
8341         */
8342         getPositioning : function(){
8343             var l = this.getStyle("left");
8344             var t = this.getStyle("top");
8345             return {
8346                 "position" : this.getStyle("position"),
8347                 "left" : l,
8348                 "right" : l ? "" : this.getStyle("right"),
8349                 "top" : t,
8350                 "bottom" : t ? "" : this.getStyle("bottom"),
8351                 "z-index" : this.getStyle("z-index")
8352             };
8353         },
8354
8355         /**
8356          * Gets the width of the border(s) for the specified side(s)
8357          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8358          * passing lr would get the border (l)eft width + the border (r)ight width.
8359          * @return {Number} The width of the sides passed added together
8360          */
8361         getBorderWidth : function(side){
8362             return this.addStyles(side, El.borders);
8363         },
8364
8365         /**
8366          * Gets the width of the padding(s) for the specified side(s)
8367          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8368          * passing lr would get the padding (l)eft + the padding (r)ight.
8369          * @return {Number} The padding of the sides passed added together
8370          */
8371         getPadding : function(side){
8372             return this.addStyles(side, El.paddings);
8373         },
8374
8375         /**
8376         * Set positioning with an object returned by getPositioning().
8377         * @param {Object} posCfg
8378         * @return {Roo.Element} this
8379          */
8380         setPositioning : function(pc){
8381             this.applyStyles(pc);
8382             if(pc.right == "auto"){
8383                 this.dom.style.right = "";
8384             }
8385             if(pc.bottom == "auto"){
8386                 this.dom.style.bottom = "";
8387             }
8388             return this;
8389         },
8390
8391         // private
8392         fixDisplay : function(){
8393             if(this.getStyle("display") == "none"){
8394                 this.setStyle("visibility", "hidden");
8395                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8396                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8397                     this.setStyle("display", "block");
8398                 }
8399             }
8400         },
8401
8402         /**
8403          * Quick set left and top adding default units
8404          * @param {String} left The left CSS property value
8405          * @param {String} top The top CSS property value
8406          * @return {Roo.Element} this
8407          */
8408          setLeftTop : function(left, top){
8409             this.dom.style.left = this.addUnits(left);
8410             this.dom.style.top = this.addUnits(top);
8411             return this;
8412         },
8413
8414         /**
8415          * Move this element relative to its current position.
8416          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8417          * @param {Number} distance How far to move the element in pixels
8418          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8419          * @return {Roo.Element} this
8420          */
8421          move : function(direction, distance, animate){
8422             var xy = this.getXY();
8423             direction = direction.toLowerCase();
8424             switch(direction){
8425                 case "l":
8426                 case "left":
8427                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8428                     break;
8429                case "r":
8430                case "right":
8431                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8432                     break;
8433                case "t":
8434                case "top":
8435                case "up":
8436                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8437                     break;
8438                case "b":
8439                case "bottom":
8440                case "down":
8441                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8442                     break;
8443             }
8444             return this;
8445         },
8446
8447         /**
8448          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8449          * @return {Roo.Element} this
8450          */
8451         clip : function(){
8452             if(!this.isClipped){
8453                this.isClipped = true;
8454                this.originalClip = {
8455                    "o": this.getStyle("overflow"),
8456                    "x": this.getStyle("overflow-x"),
8457                    "y": this.getStyle("overflow-y")
8458                };
8459                this.setStyle("overflow", "hidden");
8460                this.setStyle("overflow-x", "hidden");
8461                this.setStyle("overflow-y", "hidden");
8462             }
8463             return this;
8464         },
8465
8466         /**
8467          *  Return clipping (overflow) to original clipping before clip() was called
8468          * @return {Roo.Element} this
8469          */
8470         unclip : function(){
8471             if(this.isClipped){
8472                 this.isClipped = false;
8473                 var o = this.originalClip;
8474                 if(o.o){this.setStyle("overflow", o.o);}
8475                 if(o.x){this.setStyle("overflow-x", o.x);}
8476                 if(o.y){this.setStyle("overflow-y", o.y);}
8477             }
8478             return this;
8479         },
8480
8481
8482         /**
8483          * Gets the x,y coordinates specified by the anchor position on the element.
8484          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8485          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8486          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8487          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8488          * @return {Array} [x, y] An array containing the element's x and y coordinates
8489          */
8490         getAnchorXY : function(anchor, local, s){
8491             //Passing a different size is useful for pre-calculating anchors,
8492             //especially for anchored animations that change the el size.
8493
8494             var w, h, vp = false;
8495             if(!s){
8496                 var d = this.dom;
8497                 if(d == document.body || d == document){
8498                     vp = true;
8499                     w = D.getViewWidth(); h = D.getViewHeight();
8500                 }else{
8501                     w = this.getWidth(); h = this.getHeight();
8502                 }
8503             }else{
8504                 w = s.width;  h = s.height;
8505             }
8506             var x = 0, y = 0, r = Math.round;
8507             switch((anchor || "tl").toLowerCase()){
8508                 case "c":
8509                     x = r(w*.5);
8510                     y = r(h*.5);
8511                 break;
8512                 case "t":
8513                     x = r(w*.5);
8514                     y = 0;
8515                 break;
8516                 case "l":
8517                     x = 0;
8518                     y = r(h*.5);
8519                 break;
8520                 case "r":
8521                     x = w;
8522                     y = r(h*.5);
8523                 break;
8524                 case "b":
8525                     x = r(w*.5);
8526                     y = h;
8527                 break;
8528                 case "tl":
8529                     x = 0;
8530                     y = 0;
8531                 break;
8532                 case "bl":
8533                     x = 0;
8534                     y = h;
8535                 break;
8536                 case "br":
8537                     x = w;
8538                     y = h;
8539                 break;
8540                 case "tr":
8541                     x = w;
8542                     y = 0;
8543                 break;
8544             }
8545             if(local === true){
8546                 return [x, y];
8547             }
8548             if(vp){
8549                 var sc = this.getScroll();
8550                 return [x + sc.left, y + sc.top];
8551             }
8552             //Add the element's offset xy
8553             var o = this.getXY();
8554             return [x+o[0], y+o[1]];
8555         },
8556
8557         /**
8558          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8559          * supported position values.
8560          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8561          * @param {String} position The position to align to.
8562          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8563          * @return {Array} [x, y]
8564          */
8565         getAlignToXY : function(el, p, o){
8566             el = Roo.get(el);
8567             var d = this.dom;
8568             if(!el.dom){
8569                 throw "Element.alignTo with an element that doesn't exist";
8570             }
8571             var c = false; //constrain to viewport
8572             var p1 = "", p2 = "";
8573             o = o || [0,0];
8574
8575             if(!p){
8576                 p = "tl-bl";
8577             }else if(p == "?"){
8578                 p = "tl-bl?";
8579             }else if(p.indexOf("-") == -1){
8580                 p = "tl-" + p;
8581             }
8582             p = p.toLowerCase();
8583             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8584             if(!m){
8585                throw "Element.alignTo with an invalid alignment " + p;
8586             }
8587             p1 = m[1]; p2 = m[2]; c = !!m[3];
8588
8589             //Subtract the aligned el's internal xy from the target's offset xy
8590             //plus custom offset to get the aligned el's new offset xy
8591             var a1 = this.getAnchorXY(p1, true);
8592             var a2 = el.getAnchorXY(p2, false);
8593             var x = a2[0] - a1[0] + o[0];
8594             var y = a2[1] - a1[1] + o[1];
8595             if(c){
8596                 //constrain the aligned el to viewport if necessary
8597                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8598                 // 5px of margin for ie
8599                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8600
8601                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8602                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8603                 //otherwise swap the aligned el to the opposite border of the target.
8604                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8605                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8606                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8607                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8608
8609                var doc = document;
8610                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8611                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8612
8613                if((x+w) > dw + scrollX){
8614                     x = swapX ? r.left-w : dw+scrollX-w;
8615                 }
8616                if(x < scrollX){
8617                    x = swapX ? r.right : scrollX;
8618                }
8619                if((y+h) > dh + scrollY){
8620                     y = swapY ? r.top-h : dh+scrollY-h;
8621                 }
8622                if (y < scrollY){
8623                    y = swapY ? r.bottom : scrollY;
8624                }
8625             }
8626             return [x,y];
8627         },
8628
8629         // private
8630         getConstrainToXY : function(){
8631             var os = {top:0, left:0, bottom:0, right: 0};
8632
8633             return function(el, local, offsets, proposedXY){
8634                 el = Roo.get(el);
8635                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8636
8637                 var vw, vh, vx = 0, vy = 0;
8638                 if(el.dom == document.body || el.dom == document){
8639                     vw = Roo.lib.Dom.getViewWidth();
8640                     vh = Roo.lib.Dom.getViewHeight();
8641                 }else{
8642                     vw = el.dom.clientWidth;
8643                     vh = el.dom.clientHeight;
8644                     if(!local){
8645                         var vxy = el.getXY();
8646                         vx = vxy[0];
8647                         vy = vxy[1];
8648                     }
8649                 }
8650
8651                 var s = el.getScroll();
8652
8653                 vx += offsets.left + s.left;
8654                 vy += offsets.top + s.top;
8655
8656                 vw -= offsets.right;
8657                 vh -= offsets.bottom;
8658
8659                 var vr = vx+vw;
8660                 var vb = vy+vh;
8661
8662                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8663                 var x = xy[0], y = xy[1];
8664                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8665
8666                 // only move it if it needs it
8667                 var moved = false;
8668
8669                 // first validate right/bottom
8670                 if((x + w) > vr){
8671                     x = vr - w;
8672                     moved = true;
8673                 }
8674                 if((y + h) > vb){
8675                     y = vb - h;
8676                     moved = true;
8677                 }
8678                 // then make sure top/left isn't negative
8679                 if(x < vx){
8680                     x = vx;
8681                     moved = true;
8682                 }
8683                 if(y < vy){
8684                     y = vy;
8685                     moved = true;
8686                 }
8687                 return moved ? [x, y] : false;
8688             };
8689         }(),
8690
8691         // private
8692         adjustForConstraints : function(xy, parent, offsets){
8693             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8694         },
8695
8696         /**
8697          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8698          * document it aligns it to the viewport.
8699          * The position parameter is optional, and can be specified in any one of the following formats:
8700          * <ul>
8701          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8702          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8703          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8704          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8705          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8706          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8707          * </ul>
8708          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8709          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8710          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8711          * that specified in order to enforce the viewport constraints.
8712          * Following are all of the supported anchor positions:
8713     <pre>
8714     Value  Description
8715     -----  -----------------------------
8716     tl     The top left corner (default)
8717     t      The center of the top edge
8718     tr     The top right corner
8719     l      The center of the left edge
8720     c      In the center of the element
8721     r      The center of the right edge
8722     bl     The bottom left corner
8723     b      The center of the bottom edge
8724     br     The bottom right corner
8725     </pre>
8726     Example Usage:
8727     <pre><code>
8728     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8729     el.alignTo("other-el");
8730
8731     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8732     el.alignTo("other-el", "tr?");
8733
8734     // align the bottom right corner of el with the center left edge of other-el
8735     el.alignTo("other-el", "br-l?");
8736
8737     // align the center of el with the bottom left corner of other-el and
8738     // adjust the x position by -6 pixels (and the y position by 0)
8739     el.alignTo("other-el", "c-bl", [-6, 0]);
8740     </code></pre>
8741          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8742          * @param {String} position The position to align to.
8743          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8744          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8745          * @return {Roo.Element} this
8746          */
8747         alignTo : function(element, position, offsets, animate){
8748             var xy = this.getAlignToXY(element, position, offsets);
8749             this.setXY(xy, this.preanim(arguments, 3));
8750             return this;
8751         },
8752
8753         /**
8754          * Anchors an element to another element and realigns it when the window is resized.
8755          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8756          * @param {String} position The position to align to.
8757          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8758          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8759          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8760          * is a number, it is used as the buffer delay (defaults to 50ms).
8761          * @param {Function} callback The function to call after the animation finishes
8762          * @return {Roo.Element} this
8763          */
8764         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8765             var action = function(){
8766                 this.alignTo(el, alignment, offsets, animate);
8767                 Roo.callback(callback, this);
8768             };
8769             Roo.EventManager.onWindowResize(action, this);
8770             var tm = typeof monitorScroll;
8771             if(tm != 'undefined'){
8772                 Roo.EventManager.on(window, 'scroll', action, this,
8773                     {buffer: tm == 'number' ? monitorScroll : 50});
8774             }
8775             action.call(this); // align immediately
8776             return this;
8777         },
8778         /**
8779          * Clears any opacity settings from this element. Required in some cases for IE.
8780          * @return {Roo.Element} this
8781          */
8782         clearOpacity : function(){
8783             if (window.ActiveXObject) {
8784                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8785                     this.dom.style.filter = "";
8786                 }
8787             } else {
8788                 this.dom.style.opacity = "";
8789                 this.dom.style["-moz-opacity"] = "";
8790                 this.dom.style["-khtml-opacity"] = "";
8791             }
8792             return this;
8793         },
8794
8795         /**
8796          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8797          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8798          * @return {Roo.Element} this
8799          */
8800         hide : function(animate){
8801             this.setVisible(false, this.preanim(arguments, 0));
8802             return this;
8803         },
8804
8805         /**
8806         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8807         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8808          * @return {Roo.Element} this
8809          */
8810         show : function(animate){
8811             this.setVisible(true, this.preanim(arguments, 0));
8812             return this;
8813         },
8814
8815         /**
8816          * @private Test if size has a unit, otherwise appends the default
8817          */
8818         addUnits : function(size){
8819             return Roo.Element.addUnits(size, this.defaultUnit);
8820         },
8821
8822         /**
8823          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8824          * @return {Roo.Element} this
8825          */
8826         beginMeasure : function(){
8827             var el = this.dom;
8828             if(el.offsetWidth || el.offsetHeight){
8829                 return this; // offsets work already
8830             }
8831             var changed = [];
8832             var p = this.dom, b = document.body; // start with this element
8833             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8834                 var pe = Roo.get(p);
8835                 if(pe.getStyle('display') == 'none'){
8836                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8837                     p.style.visibility = "hidden";
8838                     p.style.display = "block";
8839                 }
8840                 p = p.parentNode;
8841             }
8842             this._measureChanged = changed;
8843             return this;
8844
8845         },
8846
8847         /**
8848          * Restores displays to before beginMeasure was called
8849          * @return {Roo.Element} this
8850          */
8851         endMeasure : function(){
8852             var changed = this._measureChanged;
8853             if(changed){
8854                 for(var i = 0, len = changed.length; i < len; i++) {
8855                     var r = changed[i];
8856                     r.el.style.visibility = r.visibility;
8857                     r.el.style.display = "none";
8858                 }
8859                 this._measureChanged = null;
8860             }
8861             return this;
8862         },
8863
8864         /**
8865         * Update the innerHTML of this element, optionally searching for and processing scripts
8866         * @param {String} html The new HTML
8867         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8868         * @param {Function} callback For async script loading you can be noticed when the update completes
8869         * @return {Roo.Element} this
8870          */
8871         update : function(html, loadScripts, callback){
8872             if(typeof html == "undefined"){
8873                 html = "";
8874             }
8875             if(loadScripts !== true){
8876                 this.dom.innerHTML = html;
8877                 if(typeof callback == "function"){
8878                     callback();
8879                 }
8880                 return this;
8881             }
8882             var id = Roo.id();
8883             var dom = this.dom;
8884
8885             html += '<span id="' + id + '"></span>';
8886
8887             E.onAvailable(id, function(){
8888                 var hd = document.getElementsByTagName("head")[0];
8889                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8890                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8891                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8892
8893                 var match;
8894                 while(match = re.exec(html)){
8895                     var attrs = match[1];
8896                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8897                     if(srcMatch && srcMatch[2]){
8898                        var s = document.createElement("script");
8899                        s.src = srcMatch[2];
8900                        var typeMatch = attrs.match(typeRe);
8901                        if(typeMatch && typeMatch[2]){
8902                            s.type = typeMatch[2];
8903                        }
8904                        hd.appendChild(s);
8905                     }else if(match[2] && match[2].length > 0){
8906                         if(window.execScript) {
8907                            window.execScript(match[2]);
8908                         } else {
8909                             /**
8910                              * eval:var:id
8911                              * eval:var:dom
8912                              * eval:var:html
8913                              * 
8914                              */
8915                            window.eval(match[2]);
8916                         }
8917                     }
8918                 }
8919                 var el = document.getElementById(id);
8920                 if(el){el.parentNode.removeChild(el);}
8921                 if(typeof callback == "function"){
8922                     callback();
8923                 }
8924             });
8925             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8926             return this;
8927         },
8928
8929         /**
8930          * Direct access to the UpdateManager update() method (takes the same parameters).
8931          * @param {String/Function} url The url for this request or a function to call to get the url
8932          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8933          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8934          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8935          * @return {Roo.Element} this
8936          */
8937         load : function(){
8938             var um = this.getUpdateManager();
8939             um.update.apply(um, arguments);
8940             return this;
8941         },
8942
8943         /**
8944         * Gets this element's UpdateManager
8945         * @return {Roo.UpdateManager} The UpdateManager
8946         */
8947         getUpdateManager : function(){
8948             if(!this.updateManager){
8949                 this.updateManager = new Roo.UpdateManager(this);
8950             }
8951             return this.updateManager;
8952         },
8953
8954         /**
8955          * Disables text selection for this element (normalized across browsers)
8956          * @return {Roo.Element} this
8957          */
8958         unselectable : function(){
8959             this.dom.unselectable = "on";
8960             this.swallowEvent("selectstart", true);
8961             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8962             this.addClass("x-unselectable");
8963             return this;
8964         },
8965
8966         /**
8967         * Calculates the x, y to center this element on the screen
8968         * @return {Array} The x, y values [x, y]
8969         */
8970         getCenterXY : function(){
8971             return this.getAlignToXY(document, 'c-c');
8972         },
8973
8974         /**
8975         * Centers the Element in either the viewport, or another Element.
8976         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8977         */
8978         center : function(centerIn){
8979             this.alignTo(centerIn || document, 'c-c');
8980             return this;
8981         },
8982
8983         /**
8984          * Tests various css rules/browsers to determine if this element uses a border box
8985          * @return {Boolean}
8986          */
8987         isBorderBox : function(){
8988             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8989         },
8990
8991         /**
8992          * Return a box {x, y, width, height} that can be used to set another elements
8993          * size/location to match this element.
8994          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8995          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8996          * @return {Object} box An object in the format {x, y, width, height}
8997          */
8998         getBox : function(contentBox, local){
8999             var xy;
9000             if(!local){
9001                 xy = this.getXY();
9002             }else{
9003                 var left = parseInt(this.getStyle("left"), 10) || 0;
9004                 var top = parseInt(this.getStyle("top"), 10) || 0;
9005                 xy = [left, top];
9006             }
9007             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9008             if(!contentBox){
9009                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9010             }else{
9011                 var l = this.getBorderWidth("l")+this.getPadding("l");
9012                 var r = this.getBorderWidth("r")+this.getPadding("r");
9013                 var t = this.getBorderWidth("t")+this.getPadding("t");
9014                 var b = this.getBorderWidth("b")+this.getPadding("b");
9015                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9016             }
9017             bx.right = bx.x + bx.width;
9018             bx.bottom = bx.y + bx.height;
9019             return bx;
9020         },
9021
9022         /**
9023          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9024          for more information about the sides.
9025          * @param {String} sides
9026          * @return {Number}
9027          */
9028         getFrameWidth : function(sides, onlyContentBox){
9029             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9030         },
9031
9032         /**
9033          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9034          * @param {Object} box The box to fill {x, y, width, height}
9035          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9036          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9037          * @return {Roo.Element} this
9038          */
9039         setBox : function(box, adjust, animate){
9040             var w = box.width, h = box.height;
9041             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9042                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9043                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9044             }
9045             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9046             return this;
9047         },
9048
9049         /**
9050          * Forces the browser to repaint this element
9051          * @return {Roo.Element} this
9052          */
9053          repaint : function(){
9054             var dom = this.dom;
9055             this.addClass("x-repaint");
9056             setTimeout(function(){
9057                 Roo.get(dom).removeClass("x-repaint");
9058             }, 1);
9059             return this;
9060         },
9061
9062         /**
9063          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9064          * then it returns the calculated width of the sides (see getPadding)
9065          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9066          * @return {Object/Number}
9067          */
9068         getMargins : function(side){
9069             if(!side){
9070                 return {
9071                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9072                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9073                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9074                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9075                 };
9076             }else{
9077                 return this.addStyles(side, El.margins);
9078              }
9079         },
9080
9081         // private
9082         addStyles : function(sides, styles){
9083             var val = 0, v, w;
9084             for(var i = 0, len = sides.length; i < len; i++){
9085                 v = this.getStyle(styles[sides.charAt(i)]);
9086                 if(v){
9087                      w = parseInt(v, 10);
9088                      if(w){ val += w; }
9089                 }
9090             }
9091             return val;
9092         },
9093
9094         /**
9095          * Creates a proxy element of this element
9096          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9097          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9098          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9099          * @return {Roo.Element} The new proxy element
9100          */
9101         createProxy : function(config, renderTo, matchBox){
9102             if(renderTo){
9103                 renderTo = Roo.getDom(renderTo);
9104             }else{
9105                 renderTo = document.body;
9106             }
9107             config = typeof config == "object" ?
9108                 config : {tag : "div", cls: config};
9109             var proxy = Roo.DomHelper.append(renderTo, config, true);
9110             if(matchBox){
9111                proxy.setBox(this.getBox());
9112             }
9113             return proxy;
9114         },
9115
9116         /**
9117          * Puts a mask over this element to disable user interaction. Requires core.css.
9118          * This method can only be applied to elements which accept child nodes.
9119          * @param {String} msg (optional) A message to display in the mask
9120          * @param {String} msgCls (optional) A css class to apply to the msg element
9121          * @return {Element} The mask  element
9122          */
9123         mask : function(msg, msgCls)
9124         {
9125             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9126                 this.setStyle("position", "relative");
9127             }
9128             if(!this._mask){
9129                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9130             }
9131             
9132             this.addClass("x-masked");
9133             this._mask.setDisplayed(true);
9134             
9135             // we wander
9136             var z = 0;
9137             var dom = this.dom;
9138             while (dom && dom.style) {
9139                 if (!isNaN(parseInt(dom.style.zIndex))) {
9140                     z = Math.max(z, parseInt(dom.style.zIndex));
9141                 }
9142                 dom = dom.parentNode;
9143             }
9144             // if we are masking the body - then it hides everything..
9145             if (this.dom == document.body) {
9146                 z = 1000000;
9147                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9148                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9149             }
9150            
9151             if(typeof msg == 'string'){
9152                 if(!this._maskMsg){
9153                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9154                         cls: "roo-el-mask-msg", 
9155                         cn: [
9156                             {
9157                                 tag: 'i',
9158                                 cls: 'fa fa-spinner fa-spin'
9159                             },
9160                             {
9161                                 tag: 'div'
9162                             }   
9163                         ]
9164                     }, true);
9165                 }
9166                 var mm = this._maskMsg;
9167                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9168                 if (mm.dom.lastChild) { // weird IE issue?
9169                     mm.dom.lastChild.innerHTML = msg;
9170                 }
9171                 mm.setDisplayed(true);
9172                 mm.center(this);
9173                 mm.setStyle('z-index', z + 102);
9174             }
9175             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9176                 this._mask.setHeight(this.getHeight());
9177             }
9178             this._mask.setStyle('z-index', z + 100);
9179             
9180             return this._mask;
9181         },
9182
9183         /**
9184          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9185          * it is cached for reuse.
9186          */
9187         unmask : function(removeEl){
9188             if(this._mask){
9189                 if(removeEl === true){
9190                     this._mask.remove();
9191                     delete this._mask;
9192                     if(this._maskMsg){
9193                         this._maskMsg.remove();
9194                         delete this._maskMsg;
9195                     }
9196                 }else{
9197                     this._mask.setDisplayed(false);
9198                     if(this._maskMsg){
9199                         this._maskMsg.setDisplayed(false);
9200                     }
9201                 }
9202             }
9203             this.removeClass("x-masked");
9204         },
9205
9206         /**
9207          * Returns true if this element is masked
9208          * @return {Boolean}
9209          */
9210         isMasked : function(){
9211             return this._mask && this._mask.isVisible();
9212         },
9213
9214         /**
9215          * Creates an iframe shim for this element to keep selects and other windowed objects from
9216          * showing through.
9217          * @return {Roo.Element} The new shim element
9218          */
9219         createShim : function(){
9220             var el = document.createElement('iframe');
9221             el.frameBorder = 'no';
9222             el.className = 'roo-shim';
9223             if(Roo.isIE && Roo.isSecure){
9224                 el.src = Roo.SSL_SECURE_URL;
9225             }
9226             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9227             shim.autoBoxAdjust = false;
9228             return shim;
9229         },
9230
9231         /**
9232          * Removes this element from the DOM and deletes it from the cache
9233          */
9234         remove : function(){
9235             if(this.dom.parentNode){
9236                 this.dom.parentNode.removeChild(this.dom);
9237             }
9238             delete El.cache[this.dom.id];
9239         },
9240
9241         /**
9242          * Sets up event handlers to add and remove a css class when the mouse is over this element
9243          * @param {String} className
9244          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9245          * mouseout events for children elements
9246          * @return {Roo.Element} this
9247          */
9248         addClassOnOver : function(className, preventFlicker){
9249             this.on("mouseover", function(){
9250                 Roo.fly(this, '_internal').addClass(className);
9251             }, this.dom);
9252             var removeFn = function(e){
9253                 if(preventFlicker !== true || !e.within(this, true)){
9254                     Roo.fly(this, '_internal').removeClass(className);
9255                 }
9256             };
9257             this.on("mouseout", removeFn, this.dom);
9258             return this;
9259         },
9260
9261         /**
9262          * Sets up event handlers to add and remove a css class when this element has the focus
9263          * @param {String} className
9264          * @return {Roo.Element} this
9265          */
9266         addClassOnFocus : function(className){
9267             this.on("focus", function(){
9268                 Roo.fly(this, '_internal').addClass(className);
9269             }, this.dom);
9270             this.on("blur", function(){
9271                 Roo.fly(this, '_internal').removeClass(className);
9272             }, this.dom);
9273             return this;
9274         },
9275         /**
9276          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9277          * @param {String} className
9278          * @return {Roo.Element} this
9279          */
9280         addClassOnClick : function(className){
9281             var dom = this.dom;
9282             this.on("mousedown", function(){
9283                 Roo.fly(dom, '_internal').addClass(className);
9284                 var d = Roo.get(document);
9285                 var fn = function(){
9286                     Roo.fly(dom, '_internal').removeClass(className);
9287                     d.removeListener("mouseup", fn);
9288                 };
9289                 d.on("mouseup", fn);
9290             });
9291             return this;
9292         },
9293
9294         /**
9295          * Stops the specified event from bubbling and optionally prevents the default action
9296          * @param {String} eventName
9297          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9298          * @return {Roo.Element} this
9299          */
9300         swallowEvent : function(eventName, preventDefault){
9301             var fn = function(e){
9302                 e.stopPropagation();
9303                 if(preventDefault){
9304                     e.preventDefault();
9305                 }
9306             };
9307             if(eventName instanceof Array){
9308                 for(var i = 0, len = eventName.length; i < len; i++){
9309                      this.on(eventName[i], fn);
9310                 }
9311                 return this;
9312             }
9313             this.on(eventName, fn);
9314             return this;
9315         },
9316
9317         /**
9318          * @private
9319          */
9320       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9321
9322         /**
9323          * Sizes this element to its parent element's dimensions performing
9324          * neccessary box adjustments.
9325          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9326          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9327          * @return {Roo.Element} this
9328          */
9329         fitToParent : function(monitorResize, targetParent) {
9330           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9331           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9332           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9333             return;
9334           }
9335           var p = Roo.get(targetParent || this.dom.parentNode);
9336           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9337           if (monitorResize === true) {
9338             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9339             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9340           }
9341           return this;
9342         },
9343
9344         /**
9345          * Gets the next sibling, skipping text nodes
9346          * @return {HTMLElement} The next sibling or null
9347          */
9348         getNextSibling : function(){
9349             var n = this.dom.nextSibling;
9350             while(n && n.nodeType != 1){
9351                 n = n.nextSibling;
9352             }
9353             return n;
9354         },
9355
9356         /**
9357          * Gets the previous sibling, skipping text nodes
9358          * @return {HTMLElement} The previous sibling or null
9359          */
9360         getPrevSibling : function(){
9361             var n = this.dom.previousSibling;
9362             while(n && n.nodeType != 1){
9363                 n = n.previousSibling;
9364             }
9365             return n;
9366         },
9367
9368
9369         /**
9370          * Appends the passed element(s) to this element
9371          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9372          * @return {Roo.Element} this
9373          */
9374         appendChild: function(el){
9375             el = Roo.get(el);
9376             el.appendTo(this);
9377             return this;
9378         },
9379
9380         /**
9381          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9382          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9383          * automatically generated with the specified attributes.
9384          * @param {HTMLElement} insertBefore (optional) a child element of this element
9385          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9386          * @return {Roo.Element} The new child element
9387          */
9388         createChild: function(config, insertBefore, returnDom){
9389             config = config || {tag:'div'};
9390             if(insertBefore){
9391                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9392             }
9393             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9394         },
9395
9396         /**
9397          * Appends this element to the passed element
9398          * @param {String/HTMLElement/Element} el The new parent element
9399          * @return {Roo.Element} this
9400          */
9401         appendTo: function(el){
9402             el = Roo.getDom(el);
9403             el.appendChild(this.dom);
9404             return this;
9405         },
9406
9407         /**
9408          * Inserts this element before the passed element in the DOM
9409          * @param {String/HTMLElement/Element} el The element to insert before
9410          * @return {Roo.Element} this
9411          */
9412         insertBefore: function(el){
9413             el = Roo.getDom(el);
9414             el.parentNode.insertBefore(this.dom, el);
9415             return this;
9416         },
9417
9418         /**
9419          * Inserts this element after the passed element in the DOM
9420          * @param {String/HTMLElement/Element} el The element to insert after
9421          * @return {Roo.Element} this
9422          */
9423         insertAfter: function(el){
9424             el = Roo.getDom(el);
9425             el.parentNode.insertBefore(this.dom, el.nextSibling);
9426             return this;
9427         },
9428
9429         /**
9430          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9431          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9432          * @return {Roo.Element} The new child
9433          */
9434         insertFirst: function(el, returnDom){
9435             el = el || {};
9436             if(typeof el == 'object' && !el.nodeType){ // dh config
9437                 return this.createChild(el, this.dom.firstChild, returnDom);
9438             }else{
9439                 el = Roo.getDom(el);
9440                 this.dom.insertBefore(el, this.dom.firstChild);
9441                 return !returnDom ? Roo.get(el) : el;
9442             }
9443         },
9444
9445         /**
9446          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9447          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9448          * @param {String} where (optional) 'before' or 'after' defaults to before
9449          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9450          * @return {Roo.Element} the inserted Element
9451          */
9452         insertSibling: function(el, where, returnDom){
9453             where = where ? where.toLowerCase() : 'before';
9454             el = el || {};
9455             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9456
9457             if(typeof el == 'object' && !el.nodeType){ // dh config
9458                 if(where == 'after' && !this.dom.nextSibling){
9459                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9460                 }else{
9461                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9462                 }
9463
9464             }else{
9465                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9466                             where == 'before' ? this.dom : this.dom.nextSibling);
9467                 if(!returnDom){
9468                     rt = Roo.get(rt);
9469                 }
9470             }
9471             return rt;
9472         },
9473
9474         /**
9475          * Creates and wraps this element with another element
9476          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9477          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9478          * @return {HTMLElement/Element} The newly created wrapper element
9479          */
9480         wrap: function(config, returnDom){
9481             if(!config){
9482                 config = {tag: "div"};
9483             }
9484             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9485             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9486             return newEl;
9487         },
9488
9489         /**
9490          * Replaces the passed element with this element
9491          * @param {String/HTMLElement/Element} el The element to replace
9492          * @return {Roo.Element} this
9493          */
9494         replace: function(el){
9495             el = Roo.get(el);
9496             this.insertBefore(el);
9497             el.remove();
9498             return this;
9499         },
9500
9501         /**
9502          * Inserts an html fragment into this element
9503          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9504          * @param {String} html The HTML fragment
9505          * @param {Boolean} returnEl True to return an Roo.Element
9506          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9507          */
9508         insertHtml : function(where, html, returnEl){
9509             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9510             return returnEl ? Roo.get(el) : el;
9511         },
9512
9513         /**
9514          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9515          * @param {Object} o The object with the attributes
9516          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9517          * @return {Roo.Element} this
9518          */
9519         set : function(o, useSet){
9520             var el = this.dom;
9521             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9522             for(var attr in o){
9523                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9524                 if(attr=="cls"){
9525                     el.className = o["cls"];
9526                 }else{
9527                     if(useSet) {
9528                         el.setAttribute(attr, o[attr]);
9529                     } else {
9530                         el[attr] = o[attr];
9531                     }
9532                 }
9533             }
9534             if(o.style){
9535                 Roo.DomHelper.applyStyles(el, o.style);
9536             }
9537             return this;
9538         },
9539
9540         /**
9541          * Convenience method for constructing a KeyMap
9542          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9543          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9544          * @param {Function} fn The function to call
9545          * @param {Object} scope (optional) The scope of the function
9546          * @return {Roo.KeyMap} The KeyMap created
9547          */
9548         addKeyListener : function(key, fn, scope){
9549             var config;
9550             if(typeof key != "object" || key instanceof Array){
9551                 config = {
9552                     key: key,
9553                     fn: fn,
9554                     scope: scope
9555                 };
9556             }else{
9557                 config = {
9558                     key : key.key,
9559                     shift : key.shift,
9560                     ctrl : key.ctrl,
9561                     alt : key.alt,
9562                     fn: fn,
9563                     scope: scope
9564                 };
9565             }
9566             return new Roo.KeyMap(this, config);
9567         },
9568
9569         /**
9570          * Creates a KeyMap for this element
9571          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9572          * @return {Roo.KeyMap} The KeyMap created
9573          */
9574         addKeyMap : function(config){
9575             return new Roo.KeyMap(this, config);
9576         },
9577
9578         /**
9579          * Returns true if this element is scrollable.
9580          * @return {Boolean}
9581          */
9582          isScrollable : function(){
9583             var dom = this.dom;
9584             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9585         },
9586
9587         /**
9588          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9589          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9590          * @param {Number} value The new scroll value
9591          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9592          * @return {Element} this
9593          */
9594
9595         scrollTo : function(side, value, animate){
9596             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9597             if(!animate || !A){
9598                 this.dom[prop] = value;
9599             }else{
9600                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9601                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9602             }
9603             return this;
9604         },
9605
9606         /**
9607          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9608          * within this element's scrollable range.
9609          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9610          * @param {Number} distance How far to scroll the element in pixels
9611          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9612          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9613          * was scrolled as far as it could go.
9614          */
9615          scroll : function(direction, distance, animate){
9616              if(!this.isScrollable()){
9617                  return;
9618              }
9619              var el = this.dom;
9620              var l = el.scrollLeft, t = el.scrollTop;
9621              var w = el.scrollWidth, h = el.scrollHeight;
9622              var cw = el.clientWidth, ch = el.clientHeight;
9623              direction = direction.toLowerCase();
9624              var scrolled = false;
9625              var a = this.preanim(arguments, 2);
9626              switch(direction){
9627                  case "l":
9628                  case "left":
9629                      if(w - l > cw){
9630                          var v = Math.min(l + distance, w-cw);
9631                          this.scrollTo("left", v, a);
9632                          scrolled = true;
9633                      }
9634                      break;
9635                 case "r":
9636                 case "right":
9637                      if(l > 0){
9638                          var v = Math.max(l - distance, 0);
9639                          this.scrollTo("left", v, a);
9640                          scrolled = true;
9641                      }
9642                      break;
9643                 case "t":
9644                 case "top":
9645                 case "up":
9646                      if(t > 0){
9647                          var v = Math.max(t - distance, 0);
9648                          this.scrollTo("top", v, a);
9649                          scrolled = true;
9650                      }
9651                      break;
9652                 case "b":
9653                 case "bottom":
9654                 case "down":
9655                      if(h - t > ch){
9656                          var v = Math.min(t + distance, h-ch);
9657                          this.scrollTo("top", v, a);
9658                          scrolled = true;
9659                      }
9660                      break;
9661              }
9662              return scrolled;
9663         },
9664
9665         /**
9666          * Translates the passed page coordinates into left/top css values for this element
9667          * @param {Number/Array} x The page x or an array containing [x, y]
9668          * @param {Number} y The page y
9669          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9670          */
9671         translatePoints : function(x, y){
9672             if(typeof x == 'object' || x instanceof Array){
9673                 y = x[1]; x = x[0];
9674             }
9675             var p = this.getStyle('position');
9676             var o = this.getXY();
9677
9678             var l = parseInt(this.getStyle('left'), 10);
9679             var t = parseInt(this.getStyle('top'), 10);
9680
9681             if(isNaN(l)){
9682                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9683             }
9684             if(isNaN(t)){
9685                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9686             }
9687
9688             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9689         },
9690
9691         /**
9692          * Returns the current scroll position of the element.
9693          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9694          */
9695         getScroll : function(){
9696             var d = this.dom, doc = document;
9697             if(d == doc || d == doc.body){
9698                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9699                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9700                 return {left: l, top: t};
9701             }else{
9702                 return {left: d.scrollLeft, top: d.scrollTop};
9703             }
9704         },
9705
9706         /**
9707          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9708          * are convert to standard 6 digit hex color.
9709          * @param {String} attr The css attribute
9710          * @param {String} defaultValue The default value to use when a valid color isn't found
9711          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9712          * YUI color anims.
9713          */
9714         getColor : function(attr, defaultValue, prefix){
9715             var v = this.getStyle(attr);
9716             if(!v || v == "transparent" || v == "inherit") {
9717                 return defaultValue;
9718             }
9719             var color = typeof prefix == "undefined" ? "#" : prefix;
9720             if(v.substr(0, 4) == "rgb("){
9721                 var rvs = v.slice(4, v.length -1).split(",");
9722                 for(var i = 0; i < 3; i++){
9723                     var h = parseInt(rvs[i]).toString(16);
9724                     if(h < 16){
9725                         h = "0" + h;
9726                     }
9727                     color += h;
9728                 }
9729             } else {
9730                 if(v.substr(0, 1) == "#"){
9731                     if(v.length == 4) {
9732                         for(var i = 1; i < 4; i++){
9733                             var c = v.charAt(i);
9734                             color +=  c + c;
9735                         }
9736                     }else if(v.length == 7){
9737                         color += v.substr(1);
9738                     }
9739                 }
9740             }
9741             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9742         },
9743
9744         /**
9745          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9746          * gradient background, rounded corners and a 4-way shadow.
9747          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9748          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9749          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9750          * @return {Roo.Element} this
9751          */
9752         boxWrap : function(cls){
9753             cls = cls || 'x-box';
9754             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9755             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9756             return el;
9757         },
9758
9759         /**
9760          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9761          * @param {String} namespace The namespace in which to look for the attribute
9762          * @param {String} name The attribute name
9763          * @return {String} The attribute value
9764          */
9765         getAttributeNS : Roo.isIE ? function(ns, name){
9766             var d = this.dom;
9767             var type = typeof d[ns+":"+name];
9768             if(type != 'undefined' && type != 'unknown'){
9769                 return d[ns+":"+name];
9770             }
9771             return d[name];
9772         } : function(ns, name){
9773             var d = this.dom;
9774             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9775         },
9776         
9777         
9778         /**
9779          * Sets or Returns the value the dom attribute value
9780          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9781          * @param {String} value (optional) The value to set the attribute to
9782          * @return {String} The attribute value
9783          */
9784         attr : function(name){
9785             if (arguments.length > 1) {
9786                 this.dom.setAttribute(name, arguments[1]);
9787                 return arguments[1];
9788             }
9789             if (typeof(name) == 'object') {
9790                 for(var i in name) {
9791                     this.attr(i, name[i]);
9792                 }
9793                 return name;
9794             }
9795             
9796             
9797             if (!this.dom.hasAttribute(name)) {
9798                 return undefined;
9799             }
9800             return this.dom.getAttribute(name);
9801         }
9802         
9803         
9804         
9805     };
9806
9807     var ep = El.prototype;
9808
9809     /**
9810      * Appends an event handler (Shorthand for addListener)
9811      * @param {String}   eventName     The type of event to append
9812      * @param {Function} fn        The method the event invokes
9813      * @param {Object} scope       (optional) The scope (this object) of the fn
9814      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9815      * @method
9816      */
9817     ep.on = ep.addListener;
9818         // backwards compat
9819     ep.mon = ep.addListener;
9820
9821     /**
9822      * Removes an event handler from this element (shorthand for removeListener)
9823      * @param {String} eventName the type of event to remove
9824      * @param {Function} fn the method the event invokes
9825      * @return {Roo.Element} this
9826      * @method
9827      */
9828     ep.un = ep.removeListener;
9829
9830     /**
9831      * true to automatically adjust width and height settings for box-model issues (default to true)
9832      */
9833     ep.autoBoxAdjust = true;
9834
9835     // private
9836     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9837
9838     // private
9839     El.addUnits = function(v, defaultUnit){
9840         if(v === "" || v == "auto"){
9841             return v;
9842         }
9843         if(v === undefined){
9844             return '';
9845         }
9846         if(typeof v == "number" || !El.unitPattern.test(v)){
9847             return v + (defaultUnit || 'px');
9848         }
9849         return v;
9850     };
9851
9852     // special markup used throughout Roo when box wrapping elements
9853     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9854     /**
9855      * Visibility mode constant - Use visibility to hide element
9856      * @static
9857      * @type Number
9858      */
9859     El.VISIBILITY = 1;
9860     /**
9861      * Visibility mode constant - Use display to hide element
9862      * @static
9863      * @type Number
9864      */
9865     El.DISPLAY = 2;
9866
9867     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9868     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9869     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9870
9871
9872
9873     /**
9874      * @private
9875      */
9876     El.cache = {};
9877
9878     var docEl;
9879
9880     /**
9881      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9882      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9883      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9884      * @return {Element} The Element object
9885      * @static
9886      */
9887     El.get = function(el){
9888         var ex, elm, id;
9889         if(!el){ return null; }
9890         if(typeof el == "string"){ // element id
9891             if(!(elm = document.getElementById(el))){
9892                 return null;
9893             }
9894             if(ex = El.cache[el]){
9895                 ex.dom = elm;
9896             }else{
9897                 ex = El.cache[el] = new El(elm);
9898             }
9899             return ex;
9900         }else if(el.tagName){ // dom element
9901             if(!(id = el.id)){
9902                 id = Roo.id(el);
9903             }
9904             if(ex = El.cache[id]){
9905                 ex.dom = el;
9906             }else{
9907                 ex = El.cache[id] = new El(el);
9908             }
9909             return ex;
9910         }else if(el instanceof El){
9911             if(el != docEl){
9912                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9913                                                               // catch case where it hasn't been appended
9914                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9915             }
9916             return el;
9917         }else if(el.isComposite){
9918             return el;
9919         }else if(el instanceof Array){
9920             return El.select(el);
9921         }else if(el == document){
9922             // create a bogus element object representing the document object
9923             if(!docEl){
9924                 var f = function(){};
9925                 f.prototype = El.prototype;
9926                 docEl = new f();
9927                 docEl.dom = document;
9928             }
9929             return docEl;
9930         }
9931         return null;
9932     };
9933
9934     // private
9935     El.uncache = function(el){
9936         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9937             if(a[i]){
9938                 delete El.cache[a[i].id || a[i]];
9939             }
9940         }
9941     };
9942
9943     // private
9944     // Garbage collection - uncache elements/purge listeners on orphaned elements
9945     // so we don't hold a reference and cause the browser to retain them
9946     El.garbageCollect = function(){
9947         if(!Roo.enableGarbageCollector){
9948             clearInterval(El.collectorThread);
9949             return;
9950         }
9951         for(var eid in El.cache){
9952             var el = El.cache[eid], d = el.dom;
9953             // -------------------------------------------------------
9954             // Determining what is garbage:
9955             // -------------------------------------------------------
9956             // !d
9957             // dom node is null, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.parentNode
9960             // no parentNode == direct orphan, definitely garbage
9961             // -------------------------------------------------------
9962             // !d.offsetParent && !document.getElementById(eid)
9963             // display none elements have no offsetParent so we will
9964             // also try to look it up by it's id. However, check
9965             // offsetParent first so we don't do unneeded lookups.
9966             // This enables collection of elements that are not orphans
9967             // directly, but somewhere up the line they have an orphan
9968             // parent.
9969             // -------------------------------------------------------
9970             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9971                 delete El.cache[eid];
9972                 if(d && Roo.enableListenerCollection){
9973                     E.purgeElement(d);
9974                 }
9975             }
9976         }
9977     }
9978     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9979
9980
9981     // dom is optional
9982     El.Flyweight = function(dom){
9983         this.dom = dom;
9984     };
9985     El.Flyweight.prototype = El.prototype;
9986
9987     El._flyweights = {};
9988     /**
9989      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9990      * the dom node can be overwritten by other code.
9991      * @param {String/HTMLElement} el The dom node or id
9992      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9993      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9994      * @static
9995      * @return {Element} The shared Element object
9996      */
9997     El.fly = function(el, named){
9998         named = named || '_global';
9999         el = Roo.getDom(el);
10000         if(!el){
10001             return null;
10002         }
10003         if(!El._flyweights[named]){
10004             El._flyweights[named] = new El.Flyweight();
10005         }
10006         El._flyweights[named].dom = el;
10007         return El._flyweights[named];
10008     };
10009
10010     /**
10011      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10012      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10013      * Shorthand of {@link Roo.Element#get}
10014      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10015      * @return {Element} The Element object
10016      * @member Roo
10017      * @method get
10018      */
10019     Roo.get = El.get;
10020     /**
10021      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10022      * the dom node can be overwritten by other code.
10023      * Shorthand of {@link Roo.Element#fly}
10024      * @param {String/HTMLElement} el The dom node or id
10025      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10026      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10027      * @static
10028      * @return {Element} The shared Element object
10029      * @member Roo
10030      * @method fly
10031      */
10032     Roo.fly = El.fly;
10033
10034     // speedy lookup for elements never to box adjust
10035     var noBoxAdjust = Roo.isStrict ? {
10036         select:1
10037     } : {
10038         input:1, select:1, textarea:1
10039     };
10040     if(Roo.isIE || Roo.isGecko){
10041         noBoxAdjust['button'] = 1;
10042     }
10043
10044
10045     Roo.EventManager.on(window, 'unload', function(){
10046         delete El.cache;
10047         delete El._flyweights;
10048     });
10049 })();
10050
10051
10052
10053
10054 if(Roo.DomQuery){
10055     Roo.Element.selectorFunction = Roo.DomQuery.select;
10056 }
10057
10058 Roo.Element.select = function(selector, unique, root){
10059     var els;
10060     if(typeof selector == "string"){
10061         els = Roo.Element.selectorFunction(selector, root);
10062     }else if(selector.length !== undefined){
10063         els = selector;
10064     }else{
10065         throw "Invalid selector";
10066     }
10067     if(unique === true){
10068         return new Roo.CompositeElement(els);
10069     }else{
10070         return new Roo.CompositeElementLite(els);
10071     }
10072 };
10073 /**
10074  * Selects elements based on the passed CSS selector to enable working on them as 1.
10075  * @param {String/Array} selector The CSS selector or an array of elements
10076  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10077  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10078  * @return {CompositeElementLite/CompositeElement}
10079  * @member Roo
10080  * @method select
10081  */
10082 Roo.select = Roo.Element.select;
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094
10095
10096
10097 /*
10098  * Based on:
10099  * Ext JS Library 1.1.1
10100  * Copyright(c) 2006-2007, Ext JS, LLC.
10101  *
10102  * Originally Released Under LGPL - original licence link has changed is not relivant.
10103  *
10104  * Fork - LGPL
10105  * <script type="text/javascript">
10106  */
10107
10108
10109
10110 //Notifies Element that fx methods are available
10111 Roo.enableFx = true;
10112
10113 /**
10114  * @class Roo.Fx
10115  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10116  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10117  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10118  * Element effects to work.</p><br/>
10119  *
10120  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10121  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10122  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10123  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10124  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10125  * expected results and should be done with care.</p><br/>
10126  *
10127  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10128  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10129 <pre>
10130 Value  Description
10131 -----  -----------------------------
10132 tl     The top left corner
10133 t      The center of the top edge
10134 tr     The top right corner
10135 l      The center of the left edge
10136 r      The center of the right edge
10137 bl     The bottom left corner
10138 b      The center of the bottom edge
10139 br     The bottom right corner
10140 </pre>
10141  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10142  * below are common options that can be passed to any Fx method.</b>
10143  * @cfg {Function} callback A function called when the effect is finished
10144  * @cfg {Object} scope The scope of the effect function
10145  * @cfg {String} easing A valid Easing value for the effect
10146  * @cfg {String} afterCls A css class to apply after the effect
10147  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10148  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10149  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10150  * effects that end with the element being visually hidden, ignored otherwise)
10151  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10152  * a function which returns such a specification that will be applied to the Element after the effect finishes
10153  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10154  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10155  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10156  */
10157 Roo.Fx = {
10158         /**
10159          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10160          * origin for the slide effect.  This function automatically handles wrapping the element with
10161          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10162          * Usage:
10163          *<pre><code>
10164 // default: slide the element in from the top
10165 el.slideIn();
10166
10167 // custom: slide the element in from the right with a 2-second duration
10168 el.slideIn('r', { duration: 2 });
10169
10170 // common config options shown with default values
10171 el.slideIn('t', {
10172     easing: 'easeOut',
10173     duration: .5
10174 });
10175 </code></pre>
10176          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10177          * @param {Object} options (optional) Object literal with any of the Fx config options
10178          * @return {Roo.Element} The Element
10179          */
10180     slideIn : function(anchor, o){
10181         var el = this.getFxEl();
10182         o = o || {};
10183
10184         el.queueFx(o, function(){
10185
10186             anchor = anchor || "t";
10187
10188             // fix display to visibility
10189             this.fixDisplay();
10190
10191             // restore values after effect
10192             var r = this.getFxRestore();
10193             var b = this.getBox();
10194             // fixed size for slide
10195             this.setSize(b);
10196
10197             // wrap if needed
10198             var wrap = this.fxWrap(r.pos, o, "hidden");
10199
10200             var st = this.dom.style;
10201             st.visibility = "visible";
10202             st.position = "absolute";
10203
10204             // clear out temp styles after slide and unwrap
10205             var after = function(){
10206                 el.fxUnwrap(wrap, r.pos, o);
10207                 st.width = r.width;
10208                 st.height = r.height;
10209                 el.afterFx(o);
10210             };
10211             // time to calc the positions
10212             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10213
10214             switch(anchor.toLowerCase()){
10215                 case "t":
10216                     wrap.setSize(b.width, 0);
10217                     st.left = st.bottom = "0";
10218                     a = {height: bh};
10219                 break;
10220                 case "l":
10221                     wrap.setSize(0, b.height);
10222                     st.right = st.top = "0";
10223                     a = {width: bw};
10224                 break;
10225                 case "r":
10226                     wrap.setSize(0, b.height);
10227                     wrap.setX(b.right);
10228                     st.left = st.top = "0";
10229                     a = {width: bw, points: pt};
10230                 break;
10231                 case "b":
10232                     wrap.setSize(b.width, 0);
10233                     wrap.setY(b.bottom);
10234                     st.left = st.top = "0";
10235                     a = {height: bh, points: pt};
10236                 break;
10237                 case "tl":
10238                     wrap.setSize(0, 0);
10239                     st.right = st.bottom = "0";
10240                     a = {width: bw, height: bh};
10241                 break;
10242                 case "bl":
10243                     wrap.setSize(0, 0);
10244                     wrap.setY(b.y+b.height);
10245                     st.right = st.top = "0";
10246                     a = {width: bw, height: bh, points: pt};
10247                 break;
10248                 case "br":
10249                     wrap.setSize(0, 0);
10250                     wrap.setXY([b.right, b.bottom]);
10251                     st.left = st.top = "0";
10252                     a = {width: bw, height: bh, points: pt};
10253                 break;
10254                 case "tr":
10255                     wrap.setSize(0, 0);
10256                     wrap.setX(b.x+b.width);
10257                     st.left = st.bottom = "0";
10258                     a = {width: bw, height: bh, points: pt};
10259                 break;
10260             }
10261             this.dom.style.visibility = "visible";
10262             wrap.show();
10263
10264             arguments.callee.anim = wrap.fxanim(a,
10265                 o,
10266                 'motion',
10267                 .5,
10268                 'easeOut', after);
10269         });
10270         return this;
10271     },
10272     
10273         /**
10274          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10275          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10276          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10277          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10278          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10279          * Usage:
10280          *<pre><code>
10281 // default: slide the element out to the top
10282 el.slideOut();
10283
10284 // custom: slide the element out to the right with a 2-second duration
10285 el.slideOut('r', { duration: 2 });
10286
10287 // common config options shown with default values
10288 el.slideOut('t', {
10289     easing: 'easeOut',
10290     duration: .5,
10291     remove: false,
10292     useDisplay: false
10293 });
10294 </code></pre>
10295          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10296          * @param {Object} options (optional) Object literal with any of the Fx config options
10297          * @return {Roo.Element} The Element
10298          */
10299     slideOut : function(anchor, o){
10300         var el = this.getFxEl();
10301         o = o || {};
10302
10303         el.queueFx(o, function(){
10304
10305             anchor = anchor || "t";
10306
10307             // restore values after effect
10308             var r = this.getFxRestore();
10309             
10310             var b = this.getBox();
10311             // fixed size for slide
10312             this.setSize(b);
10313
10314             // wrap if needed
10315             var wrap = this.fxWrap(r.pos, o, "visible");
10316
10317             var st = this.dom.style;
10318             st.visibility = "visible";
10319             st.position = "absolute";
10320
10321             wrap.setSize(b);
10322
10323             var after = function(){
10324                 if(o.useDisplay){
10325                     el.setDisplayed(false);
10326                 }else{
10327                     el.hide();
10328                 }
10329
10330                 el.fxUnwrap(wrap, r.pos, o);
10331
10332                 st.width = r.width;
10333                 st.height = r.height;
10334
10335                 el.afterFx(o);
10336             };
10337
10338             var a, zero = {to: 0};
10339             switch(anchor.toLowerCase()){
10340                 case "t":
10341                     st.left = st.bottom = "0";
10342                     a = {height: zero};
10343                 break;
10344                 case "l":
10345                     st.right = st.top = "0";
10346                     a = {width: zero};
10347                 break;
10348                 case "r":
10349                     st.left = st.top = "0";
10350                     a = {width: zero, points: {to:[b.right, b.y]}};
10351                 break;
10352                 case "b":
10353                     st.left = st.top = "0";
10354                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10355                 break;
10356                 case "tl":
10357                     st.right = st.bottom = "0";
10358                     a = {width: zero, height: zero};
10359                 break;
10360                 case "bl":
10361                     st.right = st.top = "0";
10362                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10363                 break;
10364                 case "br":
10365                     st.left = st.top = "0";
10366                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10367                 break;
10368                 case "tr":
10369                     st.left = st.bottom = "0";
10370                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10371                 break;
10372             }
10373
10374             arguments.callee.anim = wrap.fxanim(a,
10375                 o,
10376                 'motion',
10377                 .5,
10378                 "easeOut", after);
10379         });
10380         return this;
10381     },
10382
10383         /**
10384          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10385          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10386          * The element must be removed from the DOM using the 'remove' config option if desired.
10387          * Usage:
10388          *<pre><code>
10389 // default
10390 el.puff();
10391
10392 // common config options shown with default values
10393 el.puff({
10394     easing: 'easeOut',
10395     duration: .5,
10396     remove: false,
10397     useDisplay: false
10398 });
10399 </code></pre>
10400          * @param {Object} options (optional) Object literal with any of the Fx config options
10401          * @return {Roo.Element} The Element
10402          */
10403     puff : function(o){
10404         var el = this.getFxEl();
10405         o = o || {};
10406
10407         el.queueFx(o, function(){
10408             this.clearOpacity();
10409             this.show();
10410
10411             // restore values after effect
10412             var r = this.getFxRestore();
10413             var st = this.dom.style;
10414
10415             var after = function(){
10416                 if(o.useDisplay){
10417                     el.setDisplayed(false);
10418                 }else{
10419                     el.hide();
10420                 }
10421
10422                 el.clearOpacity();
10423
10424                 el.setPositioning(r.pos);
10425                 st.width = r.width;
10426                 st.height = r.height;
10427                 st.fontSize = '';
10428                 el.afterFx(o);
10429             };
10430
10431             var width = this.getWidth();
10432             var height = this.getHeight();
10433
10434             arguments.callee.anim = this.fxanim({
10435                     width : {to: this.adjustWidth(width * 2)},
10436                     height : {to: this.adjustHeight(height * 2)},
10437                     points : {by: [-(width * .5), -(height * .5)]},
10438                     opacity : {to: 0},
10439                     fontSize: {to:200, unit: "%"}
10440                 },
10441                 o,
10442                 'motion',
10443                 .5,
10444                 "easeOut", after);
10445         });
10446         return this;
10447     },
10448
10449         /**
10450          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10451          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10452          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10453          * Usage:
10454          *<pre><code>
10455 // default
10456 el.switchOff();
10457
10458 // all config options shown with default values
10459 el.switchOff({
10460     easing: 'easeIn',
10461     duration: .3,
10462     remove: false,
10463     useDisplay: false
10464 });
10465 </code></pre>
10466          * @param {Object} options (optional) Object literal with any of the Fx config options
10467          * @return {Roo.Element} The Element
10468          */
10469     switchOff : function(o){
10470         var el = this.getFxEl();
10471         o = o || {};
10472
10473         el.queueFx(o, function(){
10474             this.clearOpacity();
10475             this.clip();
10476
10477             // restore values after effect
10478             var r = this.getFxRestore();
10479             var st = this.dom.style;
10480
10481             var after = function(){
10482                 if(o.useDisplay){
10483                     el.setDisplayed(false);
10484                 }else{
10485                     el.hide();
10486                 }
10487
10488                 el.clearOpacity();
10489                 el.setPositioning(r.pos);
10490                 st.width = r.width;
10491                 st.height = r.height;
10492
10493                 el.afterFx(o);
10494             };
10495
10496             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10497                 this.clearOpacity();
10498                 (function(){
10499                     this.fxanim({
10500                         height:{to:1},
10501                         points:{by:[0, this.getHeight() * .5]}
10502                     }, o, 'motion', 0.3, 'easeIn', after);
10503                 }).defer(100, this);
10504             });
10505         });
10506         return this;
10507     },
10508
10509     /**
10510      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10511      * changed using the "attr" config option) and then fading back to the original color. If no original
10512      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10513      * Usage:
10514 <pre><code>
10515 // default: highlight background to yellow
10516 el.highlight();
10517
10518 // custom: highlight foreground text to blue for 2 seconds
10519 el.highlight("0000ff", { attr: 'color', duration: 2 });
10520
10521 // common config options shown with default values
10522 el.highlight("ffff9c", {
10523     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10524     endColor: (current color) or "ffffff",
10525     easing: 'easeIn',
10526     duration: 1
10527 });
10528 </code></pre>
10529      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10530      * @param {Object} options (optional) Object literal with any of the Fx config options
10531      * @return {Roo.Element} The Element
10532      */ 
10533     highlight : function(color, o){
10534         var el = this.getFxEl();
10535         o = o || {};
10536
10537         el.queueFx(o, function(){
10538             color = color || "ffff9c";
10539             attr = o.attr || "backgroundColor";
10540
10541             this.clearOpacity();
10542             this.show();
10543
10544             var origColor = this.getColor(attr);
10545             var restoreColor = this.dom.style[attr];
10546             endColor = (o.endColor || origColor) || "ffffff";
10547
10548             var after = function(){
10549                 el.dom.style[attr] = restoreColor;
10550                 el.afterFx(o);
10551             };
10552
10553             var a = {};
10554             a[attr] = {from: color, to: endColor};
10555             arguments.callee.anim = this.fxanim(a,
10556                 o,
10557                 'color',
10558                 1,
10559                 'easeIn', after);
10560         });
10561         return this;
10562     },
10563
10564    /**
10565     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10566     * Usage:
10567 <pre><code>
10568 // default: a single light blue ripple
10569 el.frame();
10570
10571 // custom: 3 red ripples lasting 3 seconds total
10572 el.frame("ff0000", 3, { duration: 3 });
10573
10574 // common config options shown with default values
10575 el.frame("C3DAF9", 1, {
10576     duration: 1 //duration of entire animation (not each individual ripple)
10577     // Note: Easing is not configurable and will be ignored if included
10578 });
10579 </code></pre>
10580     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10581     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10582     * @param {Object} options (optional) Object literal with any of the Fx config options
10583     * @return {Roo.Element} The Element
10584     */
10585     frame : function(color, count, o){
10586         var el = this.getFxEl();
10587         o = o || {};
10588
10589         el.queueFx(o, function(){
10590             color = color || "#C3DAF9";
10591             if(color.length == 6){
10592                 color = "#" + color;
10593             }
10594             count = count || 1;
10595             duration = o.duration || 1;
10596             this.show();
10597
10598             var b = this.getBox();
10599             var animFn = function(){
10600                 var proxy = this.createProxy({
10601
10602                      style:{
10603                         visbility:"hidden",
10604                         position:"absolute",
10605                         "z-index":"35000", // yee haw
10606                         border:"0px solid " + color
10607                      }
10608                   });
10609                 var scale = Roo.isBorderBox ? 2 : 1;
10610                 proxy.animate({
10611                     top:{from:b.y, to:b.y - 20},
10612                     left:{from:b.x, to:b.x - 20},
10613                     borderWidth:{from:0, to:10},
10614                     opacity:{from:1, to:0},
10615                     height:{from:b.height, to:(b.height + (20*scale))},
10616                     width:{from:b.width, to:(b.width + (20*scale))}
10617                 }, duration, function(){
10618                     proxy.remove();
10619                 });
10620                 if(--count > 0){
10621                      animFn.defer((duration/2)*1000, this);
10622                 }else{
10623                     el.afterFx(o);
10624                 }
10625             };
10626             animFn.call(this);
10627         });
10628         return this;
10629     },
10630
10631    /**
10632     * Creates a pause before any subsequent queued effects begin.  If there are
10633     * no effects queued after the pause it will have no effect.
10634     * Usage:
10635 <pre><code>
10636 el.pause(1);
10637 </code></pre>
10638     * @param {Number} seconds The length of time to pause (in seconds)
10639     * @return {Roo.Element} The Element
10640     */
10641     pause : function(seconds){
10642         var el = this.getFxEl();
10643         var o = {};
10644
10645         el.queueFx(o, function(){
10646             setTimeout(function(){
10647                 el.afterFx(o);
10648             }, seconds * 1000);
10649         });
10650         return this;
10651     },
10652
10653    /**
10654     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10655     * using the "endOpacity" config option.
10656     * Usage:
10657 <pre><code>
10658 // default: fade in from opacity 0 to 100%
10659 el.fadeIn();
10660
10661 // custom: fade in from opacity 0 to 75% over 2 seconds
10662 el.fadeIn({ endOpacity: .75, duration: 2});
10663
10664 // common config options shown with default values
10665 el.fadeIn({
10666     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10667     easing: 'easeOut',
10668     duration: .5
10669 });
10670 </code></pre>
10671     * @param {Object} options (optional) Object literal with any of the Fx config options
10672     * @return {Roo.Element} The Element
10673     */
10674     fadeIn : function(o){
10675         var el = this.getFxEl();
10676         o = o || {};
10677         el.queueFx(o, function(){
10678             this.setOpacity(0);
10679             this.fixDisplay();
10680             this.dom.style.visibility = 'visible';
10681             var to = o.endOpacity || 1;
10682             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10683                 o, null, .5, "easeOut", function(){
10684                 if(to == 1){
10685                     this.clearOpacity();
10686                 }
10687                 el.afterFx(o);
10688             });
10689         });
10690         return this;
10691     },
10692
10693    /**
10694     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10695     * using the "endOpacity" config option.
10696     * Usage:
10697 <pre><code>
10698 // default: fade out from the element's current opacity to 0
10699 el.fadeOut();
10700
10701 // custom: fade out from the element's current opacity to 25% over 2 seconds
10702 el.fadeOut({ endOpacity: .25, duration: 2});
10703
10704 // common config options shown with default values
10705 el.fadeOut({
10706     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10707     easing: 'easeOut',
10708     duration: .5
10709     remove: false,
10710     useDisplay: false
10711 });
10712 </code></pre>
10713     * @param {Object} options (optional) Object literal with any of the Fx config options
10714     * @return {Roo.Element} The Element
10715     */
10716     fadeOut : function(o){
10717         var el = this.getFxEl();
10718         o = o || {};
10719         el.queueFx(o, function(){
10720             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10721                 o, null, .5, "easeOut", function(){
10722                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10723                      this.dom.style.display = "none";
10724                 }else{
10725                      this.dom.style.visibility = "hidden";
10726                 }
10727                 this.clearOpacity();
10728                 el.afterFx(o);
10729             });
10730         });
10731         return this;
10732     },
10733
10734    /**
10735     * Animates the transition of an element's dimensions from a starting height/width
10736     * to an ending height/width.
10737     * Usage:
10738 <pre><code>
10739 // change height and width to 100x100 pixels
10740 el.scale(100, 100);
10741
10742 // common config options shown with default values.  The height and width will default to
10743 // the element's existing values if passed as null.
10744 el.scale(
10745     [element's width],
10746     [element's height], {
10747     easing: 'easeOut',
10748     duration: .35
10749 });
10750 </code></pre>
10751     * @param {Number} width  The new width (pass undefined to keep the original width)
10752     * @param {Number} height  The new height (pass undefined to keep the original height)
10753     * @param {Object} options (optional) Object literal with any of the Fx config options
10754     * @return {Roo.Element} The Element
10755     */
10756     scale : function(w, h, o){
10757         this.shift(Roo.apply({}, o, {
10758             width: w,
10759             height: h
10760         }));
10761         return this;
10762     },
10763
10764    /**
10765     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10766     * Any of these properties not specified in the config object will not be changed.  This effect 
10767     * requires that at least one new dimension, position or opacity setting must be passed in on
10768     * the config object in order for the function to have any effect.
10769     * Usage:
10770 <pre><code>
10771 // slide the element horizontally to x position 200 while changing the height and opacity
10772 el.shift({ x: 200, height: 50, opacity: .8 });
10773
10774 // common config options shown with default values.
10775 el.shift({
10776     width: [element's width],
10777     height: [element's height],
10778     x: [element's x position],
10779     y: [element's y position],
10780     opacity: [element's opacity],
10781     easing: 'easeOut',
10782     duration: .35
10783 });
10784 </code></pre>
10785     * @param {Object} options  Object literal with any of the Fx config options
10786     * @return {Roo.Element} The Element
10787     */
10788     shift : function(o){
10789         var el = this.getFxEl();
10790         o = o || {};
10791         el.queueFx(o, function(){
10792             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10793             if(w !== undefined){
10794                 a.width = {to: this.adjustWidth(w)};
10795             }
10796             if(h !== undefined){
10797                 a.height = {to: this.adjustHeight(h)};
10798             }
10799             if(x !== undefined || y !== undefined){
10800                 a.points = {to: [
10801                     x !== undefined ? x : this.getX(),
10802                     y !== undefined ? y : this.getY()
10803                 ]};
10804             }
10805             if(op !== undefined){
10806                 a.opacity = {to: op};
10807             }
10808             if(o.xy !== undefined){
10809                 a.points = {to: o.xy};
10810             }
10811             arguments.callee.anim = this.fxanim(a,
10812                 o, 'motion', .35, "easeOut", function(){
10813                 el.afterFx(o);
10814             });
10815         });
10816         return this;
10817     },
10818
10819         /**
10820          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10821          * ending point of the effect.
10822          * Usage:
10823          *<pre><code>
10824 // default: slide the element downward while fading out
10825 el.ghost();
10826
10827 // custom: slide the element out to the right with a 2-second duration
10828 el.ghost('r', { duration: 2 });
10829
10830 // common config options shown with default values
10831 el.ghost('b', {
10832     easing: 'easeOut',
10833     duration: .5
10834     remove: false,
10835     useDisplay: false
10836 });
10837 </code></pre>
10838          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10839          * @param {Object} options (optional) Object literal with any of the Fx config options
10840          * @return {Roo.Element} The Element
10841          */
10842     ghost : function(anchor, o){
10843         var el = this.getFxEl();
10844         o = o || {};
10845
10846         el.queueFx(o, function(){
10847             anchor = anchor || "b";
10848
10849             // restore values after effect
10850             var r = this.getFxRestore();
10851             var w = this.getWidth(),
10852                 h = this.getHeight();
10853
10854             var st = this.dom.style;
10855
10856             var after = function(){
10857                 if(o.useDisplay){
10858                     el.setDisplayed(false);
10859                 }else{
10860                     el.hide();
10861                 }
10862
10863                 el.clearOpacity();
10864                 el.setPositioning(r.pos);
10865                 st.width = r.width;
10866                 st.height = r.height;
10867
10868                 el.afterFx(o);
10869             };
10870
10871             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10872             switch(anchor.toLowerCase()){
10873                 case "t":
10874                     pt.by = [0, -h];
10875                 break;
10876                 case "l":
10877                     pt.by = [-w, 0];
10878                 break;
10879                 case "r":
10880                     pt.by = [w, 0];
10881                 break;
10882                 case "b":
10883                     pt.by = [0, h];
10884                 break;
10885                 case "tl":
10886                     pt.by = [-w, -h];
10887                 break;
10888                 case "bl":
10889                     pt.by = [-w, h];
10890                 break;
10891                 case "br":
10892                     pt.by = [w, h];
10893                 break;
10894                 case "tr":
10895                     pt.by = [w, -h];
10896                 break;
10897             }
10898
10899             arguments.callee.anim = this.fxanim(a,
10900                 o,
10901                 'motion',
10902                 .5,
10903                 "easeOut", after);
10904         });
10905         return this;
10906     },
10907
10908         /**
10909          * Ensures that all effects queued after syncFx is called on the element are
10910          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10911          * @return {Roo.Element} The Element
10912          */
10913     syncFx : function(){
10914         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10915             block : false,
10916             concurrent : true,
10917             stopFx : false
10918         });
10919         return this;
10920     },
10921
10922         /**
10923          * Ensures that all effects queued after sequenceFx is called on the element are
10924          * run in sequence.  This is the opposite of {@link #syncFx}.
10925          * @return {Roo.Element} The Element
10926          */
10927     sequenceFx : function(){
10928         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10929             block : false,
10930             concurrent : false,
10931             stopFx : false
10932         });
10933         return this;
10934     },
10935
10936         /* @private */
10937     nextFx : function(){
10938         var ef = this.fxQueue[0];
10939         if(ef){
10940             ef.call(this);
10941         }
10942     },
10943
10944         /**
10945          * Returns true if the element has any effects actively running or queued, else returns false.
10946          * @return {Boolean} True if element has active effects, else false
10947          */
10948     hasActiveFx : function(){
10949         return this.fxQueue && this.fxQueue[0];
10950     },
10951
10952         /**
10953          * Stops any running effects and clears the element's internal effects queue if it contains
10954          * any additional effects that haven't started yet.
10955          * @return {Roo.Element} The Element
10956          */
10957     stopFx : function(){
10958         if(this.hasActiveFx()){
10959             var cur = this.fxQueue[0];
10960             if(cur && cur.anim && cur.anim.isAnimated()){
10961                 this.fxQueue = [cur]; // clear out others
10962                 cur.anim.stop(true);
10963             }
10964         }
10965         return this;
10966     },
10967
10968         /* @private */
10969     beforeFx : function(o){
10970         if(this.hasActiveFx() && !o.concurrent){
10971            if(o.stopFx){
10972                this.stopFx();
10973                return true;
10974            }
10975            return false;
10976         }
10977         return true;
10978     },
10979
10980         /**
10981          * Returns true if the element is currently blocking so that no other effect can be queued
10982          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10983          * used to ensure that an effect initiated by a user action runs to completion prior to the
10984          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10985          * @return {Boolean} True if blocking, else false
10986          */
10987     hasFxBlock : function(){
10988         var q = this.fxQueue;
10989         return q && q[0] && q[0].block;
10990     },
10991
10992         /* @private */
10993     queueFx : function(o, fn){
10994         if(!this.fxQueue){
10995             this.fxQueue = [];
10996         }
10997         if(!this.hasFxBlock()){
10998             Roo.applyIf(o, this.fxDefaults);
10999             if(!o.concurrent){
11000                 var run = this.beforeFx(o);
11001                 fn.block = o.block;
11002                 this.fxQueue.push(fn);
11003                 if(run){
11004                     this.nextFx();
11005                 }
11006             }else{
11007                 fn.call(this);
11008             }
11009         }
11010         return this;
11011     },
11012
11013         /* @private */
11014     fxWrap : function(pos, o, vis){
11015         var wrap;
11016         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11017             var wrapXY;
11018             if(o.fixPosition){
11019                 wrapXY = this.getXY();
11020             }
11021             var div = document.createElement("div");
11022             div.style.visibility = vis;
11023             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11024             wrap.setPositioning(pos);
11025             if(wrap.getStyle("position") == "static"){
11026                 wrap.position("relative");
11027             }
11028             this.clearPositioning('auto');
11029             wrap.clip();
11030             wrap.dom.appendChild(this.dom);
11031             if(wrapXY){
11032                 wrap.setXY(wrapXY);
11033             }
11034         }
11035         return wrap;
11036     },
11037
11038         /* @private */
11039     fxUnwrap : function(wrap, pos, o){
11040         this.clearPositioning();
11041         this.setPositioning(pos);
11042         if(!o.wrap){
11043             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11044             wrap.remove();
11045         }
11046     },
11047
11048         /* @private */
11049     getFxRestore : function(){
11050         var st = this.dom.style;
11051         return {pos: this.getPositioning(), width: st.width, height : st.height};
11052     },
11053
11054         /* @private */
11055     afterFx : function(o){
11056         if(o.afterStyle){
11057             this.applyStyles(o.afterStyle);
11058         }
11059         if(o.afterCls){
11060             this.addClass(o.afterCls);
11061         }
11062         if(o.remove === true){
11063             this.remove();
11064         }
11065         Roo.callback(o.callback, o.scope, [this]);
11066         if(!o.concurrent){
11067             this.fxQueue.shift();
11068             this.nextFx();
11069         }
11070     },
11071
11072         /* @private */
11073     getFxEl : function(){ // support for composite element fx
11074         return Roo.get(this.dom);
11075     },
11076
11077         /* @private */
11078     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11079         animType = animType || 'run';
11080         opt = opt || {};
11081         var anim = Roo.lib.Anim[animType](
11082             this.dom, args,
11083             (opt.duration || defaultDur) || .35,
11084             (opt.easing || defaultEase) || 'easeOut',
11085             function(){
11086                 Roo.callback(cb, this);
11087             },
11088             this
11089         );
11090         opt.anim = anim;
11091         return anim;
11092     }
11093 };
11094
11095 // backwords compat
11096 Roo.Fx.resize = Roo.Fx.scale;
11097
11098 //When included, Roo.Fx is automatically applied to Element so that all basic
11099 //effects are available directly via the Element API
11100 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11101  * Based on:
11102  * Ext JS Library 1.1.1
11103  * Copyright(c) 2006-2007, Ext JS, LLC.
11104  *
11105  * Originally Released Under LGPL - original licence link has changed is not relivant.
11106  *
11107  * Fork - LGPL
11108  * <script type="text/javascript">
11109  */
11110
11111
11112 /**
11113  * @class Roo.CompositeElement
11114  * Standard composite class. Creates a Roo.Element for every element in the collection.
11115  * <br><br>
11116  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11117  * actions will be performed on all the elements in this collection.</b>
11118  * <br><br>
11119  * All methods return <i>this</i> and can be chained.
11120  <pre><code>
11121  var els = Roo.select("#some-el div.some-class", true);
11122  // or select directly from an existing element
11123  var el = Roo.get('some-el');
11124  el.select('div.some-class', true);
11125
11126  els.setWidth(100); // all elements become 100 width
11127  els.hide(true); // all elements fade out and hide
11128  // or
11129  els.setWidth(100).hide(true);
11130  </code></pre>
11131  */
11132 Roo.CompositeElement = function(els){
11133     this.elements = [];
11134     this.addElements(els);
11135 };
11136 Roo.CompositeElement.prototype = {
11137     isComposite: true,
11138     addElements : function(els){
11139         if(!els) {
11140             return this;
11141         }
11142         if(typeof els == "string"){
11143             els = Roo.Element.selectorFunction(els);
11144         }
11145         var yels = this.elements;
11146         var index = yels.length-1;
11147         for(var i = 0, len = els.length; i < len; i++) {
11148                 yels[++index] = Roo.get(els[i]);
11149         }
11150         return this;
11151     },
11152
11153     /**
11154     * Clears this composite and adds the elements returned by the passed selector.
11155     * @param {String/Array} els A string CSS selector, an array of elements or an element
11156     * @return {CompositeElement} this
11157     */
11158     fill : function(els){
11159         this.elements = [];
11160         this.add(els);
11161         return this;
11162     },
11163
11164     /**
11165     * Filters this composite to only elements that match the passed selector.
11166     * @param {String} selector A string CSS selector
11167     * @param {Boolean} inverse return inverse filter (not matches)
11168     * @return {CompositeElement} this
11169     */
11170     filter : function(selector, inverse){
11171         var els = [];
11172         inverse = inverse || false;
11173         this.each(function(el){
11174             var match = inverse ? !el.is(selector) : el.is(selector);
11175             if(match){
11176                 els[els.length] = el.dom;
11177             }
11178         });
11179         this.fill(els);
11180         return this;
11181     },
11182
11183     invoke : function(fn, args){
11184         var els = this.elements;
11185         for(var i = 0, len = els.length; i < len; i++) {
11186                 Roo.Element.prototype[fn].apply(els[i], args);
11187         }
11188         return this;
11189     },
11190     /**
11191     * Adds elements to this composite.
11192     * @param {String/Array} els A string CSS selector, an array of elements or an element
11193     * @return {CompositeElement} this
11194     */
11195     add : function(els){
11196         if(typeof els == "string"){
11197             this.addElements(Roo.Element.selectorFunction(els));
11198         }else if(els.length !== undefined){
11199             this.addElements(els);
11200         }else{
11201             this.addElements([els]);
11202         }
11203         return this;
11204     },
11205     /**
11206     * Calls the passed function passing (el, this, index) for each element in this composite.
11207     * @param {Function} fn The function to call
11208     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11209     * @return {CompositeElement} this
11210     */
11211     each : function(fn, scope){
11212         var els = this.elements;
11213         for(var i = 0, len = els.length; i < len; i++){
11214             if(fn.call(scope || els[i], els[i], this, i) === false) {
11215                 break;
11216             }
11217         }
11218         return this;
11219     },
11220
11221     /**
11222      * Returns the Element object at the specified index
11223      * @param {Number} index
11224      * @return {Roo.Element}
11225      */
11226     item : function(index){
11227         return this.elements[index] || null;
11228     },
11229
11230     /**
11231      * Returns the first Element
11232      * @return {Roo.Element}
11233      */
11234     first : function(){
11235         return this.item(0);
11236     },
11237
11238     /**
11239      * Returns the last Element
11240      * @return {Roo.Element}
11241      */
11242     last : function(){
11243         return this.item(this.elements.length-1);
11244     },
11245
11246     /**
11247      * Returns the number of elements in this composite
11248      * @return Number
11249      */
11250     getCount : function(){
11251         return this.elements.length;
11252     },
11253
11254     /**
11255      * Returns true if this composite contains the passed element
11256      * @return Boolean
11257      */
11258     contains : function(el){
11259         return this.indexOf(el) !== -1;
11260     },
11261
11262     /**
11263      * Returns true if this composite contains the passed element
11264      * @return Boolean
11265      */
11266     indexOf : function(el){
11267         return this.elements.indexOf(Roo.get(el));
11268     },
11269
11270
11271     /**
11272     * Removes the specified element(s).
11273     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11274     * or an array of any of those.
11275     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11276     * @return {CompositeElement} this
11277     */
11278     removeElement : function(el, removeDom){
11279         if(el instanceof Array){
11280             for(var i = 0, len = el.length; i < len; i++){
11281                 this.removeElement(el[i]);
11282             }
11283             return this;
11284         }
11285         var index = typeof el == 'number' ? el : this.indexOf(el);
11286         if(index !== -1){
11287             if(removeDom){
11288                 var d = this.elements[index];
11289                 if(d.dom){
11290                     d.remove();
11291                 }else{
11292                     d.parentNode.removeChild(d);
11293                 }
11294             }
11295             this.elements.splice(index, 1);
11296         }
11297         return this;
11298     },
11299
11300     /**
11301     * Replaces the specified element with the passed element.
11302     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11303     * to replace.
11304     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11305     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11306     * @return {CompositeElement} this
11307     */
11308     replaceElement : function(el, replacement, domReplace){
11309         var index = typeof el == 'number' ? el : this.indexOf(el);
11310         if(index !== -1){
11311             if(domReplace){
11312                 this.elements[index].replaceWith(replacement);
11313             }else{
11314                 this.elements.splice(index, 1, Roo.get(replacement))
11315             }
11316         }
11317         return this;
11318     },
11319
11320     /**
11321      * Removes all elements.
11322      */
11323     clear : function(){
11324         this.elements = [];
11325     }
11326 };
11327 (function(){
11328     Roo.CompositeElement.createCall = function(proto, fnName){
11329         if(!proto[fnName]){
11330             proto[fnName] = function(){
11331                 return this.invoke(fnName, arguments);
11332             };
11333         }
11334     };
11335     for(var fnName in Roo.Element.prototype){
11336         if(typeof Roo.Element.prototype[fnName] == "function"){
11337             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11338         }
11339     };
11340 })();
11341 /*
11342  * Based on:
11343  * Ext JS Library 1.1.1
11344  * Copyright(c) 2006-2007, Ext JS, LLC.
11345  *
11346  * Originally Released Under LGPL - original licence link has changed is not relivant.
11347  *
11348  * Fork - LGPL
11349  * <script type="text/javascript">
11350  */
11351
11352 /**
11353  * @class Roo.CompositeElementLite
11354  * @extends Roo.CompositeElement
11355  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11356  <pre><code>
11357  var els = Roo.select("#some-el div.some-class");
11358  // or select directly from an existing element
11359  var el = Roo.get('some-el');
11360  el.select('div.some-class');
11361
11362  els.setWidth(100); // all elements become 100 width
11363  els.hide(true); // all elements fade out and hide
11364  // or
11365  els.setWidth(100).hide(true);
11366  </code></pre><br><br>
11367  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11368  * actions will be performed on all the elements in this collection.</b>
11369  */
11370 Roo.CompositeElementLite = function(els){
11371     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11372     this.el = new Roo.Element.Flyweight();
11373 };
11374 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11375     addElements : function(els){
11376         if(els){
11377             if(els instanceof Array){
11378                 this.elements = this.elements.concat(els);
11379             }else{
11380                 var yels = this.elements;
11381                 var index = yels.length-1;
11382                 for(var i = 0, len = els.length; i < len; i++) {
11383                     yels[++index] = els[i];
11384                 }
11385             }
11386         }
11387         return this;
11388     },
11389     invoke : function(fn, args){
11390         var els = this.elements;
11391         var el = this.el;
11392         for(var i = 0, len = els.length; i < len; i++) {
11393             el.dom = els[i];
11394                 Roo.Element.prototype[fn].apply(el, args);
11395         }
11396         return this;
11397     },
11398     /**
11399      * Returns a flyweight Element of the dom element object at the specified index
11400      * @param {Number} index
11401      * @return {Roo.Element}
11402      */
11403     item : function(index){
11404         if(!this.elements[index]){
11405             return null;
11406         }
11407         this.el.dom = this.elements[index];
11408         return this.el;
11409     },
11410
11411     // fixes scope with flyweight
11412     addListener : function(eventName, handler, scope, opt){
11413         var els = this.elements;
11414         for(var i = 0, len = els.length; i < len; i++) {
11415             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11416         }
11417         return this;
11418     },
11419
11420     /**
11421     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11422     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11423     * a reference to the dom node, use el.dom.</b>
11424     * @param {Function} fn The function to call
11425     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11426     * @return {CompositeElement} this
11427     */
11428     each : function(fn, scope){
11429         var els = this.elements;
11430         var el = this.el;
11431         for(var i = 0, len = els.length; i < len; i++){
11432             el.dom = els[i];
11433                 if(fn.call(scope || el, el, this, i) === false){
11434                 break;
11435             }
11436         }
11437         return this;
11438     },
11439
11440     indexOf : function(el){
11441         return this.elements.indexOf(Roo.getDom(el));
11442     },
11443
11444     replaceElement : function(el, replacement, domReplace){
11445         var index = typeof el == 'number' ? el : this.indexOf(el);
11446         if(index !== -1){
11447             replacement = Roo.getDom(replacement);
11448             if(domReplace){
11449                 var d = this.elements[index];
11450                 d.parentNode.insertBefore(replacement, d);
11451                 d.parentNode.removeChild(d);
11452             }
11453             this.elements.splice(index, 1, replacement);
11454         }
11455         return this;
11456     }
11457 });
11458 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11459
11460 /*
11461  * Based on:
11462  * Ext JS Library 1.1.1
11463  * Copyright(c) 2006-2007, Ext JS, LLC.
11464  *
11465  * Originally Released Under LGPL - original licence link has changed is not relivant.
11466  *
11467  * Fork - LGPL
11468  * <script type="text/javascript">
11469  */
11470
11471  
11472
11473 /**
11474  * @class Roo.data.Connection
11475  * @extends Roo.util.Observable
11476  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11477  * either to a configured URL, or to a URL specified at request time. 
11478  * 
11479  * Requests made by this class are asynchronous, and will return immediately. No data from
11480  * the server will be available to the statement immediately following the {@link #request} call.
11481  * To process returned data, use a callback in the request options object, or an event listener.
11482  * 
11483  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11484  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11485  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11486  * property and, if present, the IFRAME's XML document as the responseXML property.
11487  * 
11488  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11489  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11490  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11491  * standard DOM methods.
11492  * @constructor
11493  * @param {Object} config a configuration object.
11494  */
11495 Roo.data.Connection = function(config){
11496     Roo.apply(this, config);
11497     this.addEvents({
11498         /**
11499          * @event beforerequest
11500          * Fires before a network request is made to retrieve a data object.
11501          * @param {Connection} conn This Connection object.
11502          * @param {Object} options The options config object passed to the {@link #request} method.
11503          */
11504         "beforerequest" : true,
11505         /**
11506          * @event requestcomplete
11507          * Fires if the request was successfully completed.
11508          * @param {Connection} conn This Connection object.
11509          * @param {Object} response The XHR object containing the response data.
11510          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11511          * @param {Object} options The options config object passed to the {@link #request} method.
11512          */
11513         "requestcomplete" : true,
11514         /**
11515          * @event requestexception
11516          * Fires if an error HTTP status was returned from the server.
11517          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11518          * @param {Connection} conn This Connection object.
11519          * @param {Object} response The XHR object containing the response data.
11520          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11521          * @param {Object} options The options config object passed to the {@link #request} method.
11522          */
11523         "requestexception" : true
11524     });
11525     Roo.data.Connection.superclass.constructor.call(this);
11526 };
11527
11528 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11529     /**
11530      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11531      */
11532     /**
11533      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11534      * extra parameters to each request made by this object. (defaults to undefined)
11535      */
11536     /**
11537      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11538      *  to each request made by this object. (defaults to undefined)
11539      */
11540     /**
11541      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11542      */
11543     /**
11544      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11545      */
11546     timeout : 30000,
11547     /**
11548      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11549      * @type Boolean
11550      */
11551     autoAbort:false,
11552
11553     /**
11554      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11555      * @type Boolean
11556      */
11557     disableCaching: true,
11558
11559     /**
11560      * Sends an HTTP request to a remote server.
11561      * @param {Object} options An object which may contain the following properties:<ul>
11562      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11563      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11564      * request, a url encoded string or a function to call to get either.</li>
11565      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11566      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11567      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11568      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11569      * <li>options {Object} The parameter to the request call.</li>
11570      * <li>success {Boolean} True if the request succeeded.</li>
11571      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11572      * </ul></li>
11573      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11574      * The callback is passed the following parameters:<ul>
11575      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11576      * <li>options {Object} The parameter to the request call.</li>
11577      * </ul></li>
11578      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11579      * The callback is passed the following parameters:<ul>
11580      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11581      * <li>options {Object} The parameter to the request call.</li>
11582      * </ul></li>
11583      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11584      * for the callback function. Defaults to the browser window.</li>
11585      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11586      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11587      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11588      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11589      * params for the post data. Any params will be appended to the URL.</li>
11590      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11591      * </ul>
11592      * @return {Number} transactionId
11593      */
11594     request : function(o){
11595         if(this.fireEvent("beforerequest", this, o) !== false){
11596             var p = o.params;
11597
11598             if(typeof p == "function"){
11599                 p = p.call(o.scope||window, o);
11600             }
11601             if(typeof p == "object"){
11602                 p = Roo.urlEncode(o.params);
11603             }
11604             if(this.extraParams){
11605                 var extras = Roo.urlEncode(this.extraParams);
11606                 p = p ? (p + '&' + extras) : extras;
11607             }
11608
11609             var url = o.url || this.url;
11610             if(typeof url == 'function'){
11611                 url = url.call(o.scope||window, o);
11612             }
11613
11614             if(o.form){
11615                 var form = Roo.getDom(o.form);
11616                 url = url || form.action;
11617
11618                 var enctype = form.getAttribute("enctype");
11619                 
11620                 if (o.formData) {
11621                     return this.doFormDataUpload(o,p,url);
11622                 }
11623                 
11624                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11625                     return this.doFormUpload(o, p, url);
11626                 }
11627                 var f = Roo.lib.Ajax.serializeForm(form);
11628                 p = p ? (p + '&' + f) : f;
11629             }
11630
11631             var hs = o.headers;
11632             if(this.defaultHeaders){
11633                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11634                 if(!o.headers){
11635                     o.headers = hs;
11636                 }
11637             }
11638
11639             var cb = {
11640                 success: this.handleResponse,
11641                 failure: this.handleFailure,
11642                 scope: this,
11643                 argument: {options: o},
11644                 timeout : o.timeout || this.timeout
11645             };
11646
11647             var method = o.method||this.method||(p ? "POST" : "GET");
11648
11649             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11650                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11651             }
11652
11653             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11654                 if(o.autoAbort){
11655                     this.abort();
11656                 }
11657             }else if(this.autoAbort !== false){
11658                 this.abort();
11659             }
11660
11661             if((method == 'GET' && p) || o.xmlData){
11662                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11663                 p = '';
11664             }
11665             Roo.lib.Ajax.useDefaultHeader = typeof(o.headers) == 'undefined' || typeof(o.headers['Content-Type']) == 'undefined';
11666             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11667             Roo.lib.Ajax.useDefaultHeader == true;
11668             return this.transId;
11669         }else{
11670             Roo.callback(o.callback, o.scope, [o, null, null]);
11671             return null;
11672         }
11673     },
11674
11675     /**
11676      * Determine whether this object has a request outstanding.
11677      * @param {Number} transactionId (Optional) defaults to the last transaction
11678      * @return {Boolean} True if there is an outstanding request.
11679      */
11680     isLoading : function(transId){
11681         if(transId){
11682             return Roo.lib.Ajax.isCallInProgress(transId);
11683         }else{
11684             return this.transId ? true : false;
11685         }
11686     },
11687
11688     /**
11689      * Aborts any outstanding request.
11690      * @param {Number} transactionId (Optional) defaults to the last transaction
11691      */
11692     abort : function(transId){
11693         if(transId || this.isLoading()){
11694             Roo.lib.Ajax.abort(transId || this.transId);
11695         }
11696     },
11697
11698     // private
11699     handleResponse : function(response){
11700         this.transId = false;
11701         var options = response.argument.options;
11702         response.argument = options ? options.argument : null;
11703         this.fireEvent("requestcomplete", this, response, options);
11704         Roo.callback(options.success, options.scope, [response, options]);
11705         Roo.callback(options.callback, options.scope, [options, true, response]);
11706     },
11707
11708     // private
11709     handleFailure : function(response, e){
11710         this.transId = false;
11711         var options = response.argument.options;
11712         response.argument = options ? options.argument : null;
11713         this.fireEvent("requestexception", this, response, options, e);
11714         Roo.callback(options.failure, options.scope, [response, options]);
11715         Roo.callback(options.callback, options.scope, [options, false, response]);
11716     },
11717
11718     // private
11719     doFormUpload : function(o, ps, url){
11720         var id = Roo.id();
11721         var frame = document.createElement('iframe');
11722         frame.id = id;
11723         frame.name = id;
11724         frame.className = 'x-hidden';
11725         if(Roo.isIE){
11726             frame.src = Roo.SSL_SECURE_URL;
11727         }
11728         document.body.appendChild(frame);
11729
11730         if(Roo.isIE){
11731            document.frames[id].name = id;
11732         }
11733
11734         var form = Roo.getDom(o.form);
11735         form.target = id;
11736         form.method = 'POST';
11737         form.enctype = form.encoding = 'multipart/form-data';
11738         if(url){
11739             form.action = url;
11740         }
11741
11742         var hiddens, hd;
11743         if(ps){ // add dynamic params
11744             hiddens = [];
11745             ps = Roo.urlDecode(ps, false);
11746             for(var k in ps){
11747                 if(ps.hasOwnProperty(k)){
11748                     hd = document.createElement('input');
11749                     hd.type = 'hidden';
11750                     hd.name = k;
11751                     hd.value = ps[k];
11752                     form.appendChild(hd);
11753                     hiddens.push(hd);
11754                 }
11755             }
11756         }
11757
11758         function cb(){
11759             var r = {  // bogus response object
11760                 responseText : '',
11761                 responseXML : null
11762             };
11763
11764             r.argument = o ? o.argument : null;
11765
11766             try { //
11767                 var doc;
11768                 if(Roo.isIE){
11769                     doc = frame.contentWindow.document;
11770                 }else {
11771                     doc = (frame.contentDocument || window.frames[id].document);
11772                 }
11773                 if(doc && doc.body){
11774                     r.responseText = doc.body.innerHTML;
11775                 }
11776                 if(doc && doc.XMLDocument){
11777                     r.responseXML = doc.XMLDocument;
11778                 }else {
11779                     r.responseXML = doc;
11780                 }
11781             }
11782             catch(e) {
11783                 // ignore
11784             }
11785
11786             Roo.EventManager.removeListener(frame, 'load', cb, this);
11787
11788             this.fireEvent("requestcomplete", this, r, o);
11789             Roo.callback(o.success, o.scope, [r, o]);
11790             Roo.callback(o.callback, o.scope, [o, true, r]);
11791
11792             setTimeout(function(){document.body.removeChild(frame);}, 100);
11793         }
11794
11795         Roo.EventManager.on(frame, 'load', cb, this);
11796         form.submit();
11797
11798         if(hiddens){ // remove dynamic params
11799             for(var i = 0, len = hiddens.length; i < len; i++){
11800                 form.removeChild(hiddens[i]);
11801             }
11802         }
11803     },
11804     // this is a 'formdata version???'
11805     
11806     
11807     doFormDataUpload : function(o, ps, url)
11808     {
11809         var form = Roo.getDom(o.form);
11810         form.enctype = form.encoding = 'multipart/form-data';
11811         var formData = o.formData === true ? new FormData(form) : o.formData;
11812       
11813         var cb = {
11814             success: this.handleResponse,
11815             failure: this.handleFailure,
11816             scope: this,
11817             argument: {options: o},
11818             timeout : o.timeout || this.timeout
11819         };
11820  
11821         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11822             if(o.autoAbort){
11823                 this.abort();
11824             }
11825         }else if(this.autoAbort !== false){
11826             this.abort();
11827         }
11828
11829         //Roo.lib.Ajax.defaultPostHeader = null;
11830         Roo.lib.Ajax.useDefaultHeader = false;
11831         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11832         Roo.lib.Ajax.useDefaultHeader = true;
11833  
11834          
11835     }
11836     
11837 });
11838 /*
11839  * Based on:
11840  * Ext JS Library 1.1.1
11841  * Copyright(c) 2006-2007, Ext JS, LLC.
11842  *
11843  * Originally Released Under LGPL - original licence link has changed is not relivant.
11844  *
11845  * Fork - LGPL
11846  * <script type="text/javascript">
11847  */
11848  
11849 /**
11850  * Global Ajax request class.
11851  * 
11852  * @class Roo.Ajax
11853  * @extends Roo.data.Connection
11854  * @static
11855  * 
11856  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11857  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11858  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11859  * @cfg {String} method (Optional)  The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11860  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11861  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11862  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11863  */
11864 Roo.Ajax = new Roo.data.Connection({
11865     // fix up the docs
11866     /**
11867      * @scope Roo.Ajax
11868      * @type {Boolear} 
11869      */
11870     autoAbort : false,
11871
11872     /**
11873      * Serialize the passed form into a url encoded string
11874      * @scope Roo.Ajax
11875      * @param {String/HTMLElement} form
11876      * @return {String}
11877      */
11878     serializeForm : function(form){
11879         return Roo.lib.Ajax.serializeForm(form);
11880     }
11881 });/*
11882  * Based on:
11883  * Ext JS Library 1.1.1
11884  * Copyright(c) 2006-2007, Ext JS, LLC.
11885  *
11886  * Originally Released Under LGPL - original licence link has changed is not relivant.
11887  *
11888  * Fork - LGPL
11889  * <script type="text/javascript">
11890  */
11891
11892  
11893 /**
11894  * @class Roo.UpdateManager
11895  * @extends Roo.util.Observable
11896  * Provides AJAX-style update for Element object.<br><br>
11897  * Usage:<br>
11898  * <pre><code>
11899  * // Get it from a Roo.Element object
11900  * var el = Roo.get("foo");
11901  * var mgr = el.getUpdateManager();
11902  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11903  * ...
11904  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11905  * <br>
11906  * // or directly (returns the same UpdateManager instance)
11907  * var mgr = new Roo.UpdateManager("myElementId");
11908  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11909  * mgr.on("update", myFcnNeedsToKnow);
11910  * <br>
11911    // short handed call directly from the element object
11912    Roo.get("foo").load({
11913         url: "bar.php",
11914         scripts:true,
11915         params: "for=bar",
11916         text: "Loading Foo..."
11917    });
11918  * </code></pre>
11919  * @constructor
11920  * Create new UpdateManager directly.
11921  * @param {String/HTMLElement/Roo.Element} el The element to update
11922  * @param {Boolean} forceNew (optional) By default the constructor checks to see if the passed element already has an UpdateManager and if it does it returns the same instance. This will skip that check (useful for extending this class).
11923  */
11924 Roo.UpdateManager = function(el, forceNew){
11925     el = Roo.get(el);
11926     if(!forceNew && el.updateManager){
11927         return el.updateManager;
11928     }
11929     /**
11930      * The Element object
11931      * @type Roo.Element
11932      */
11933     this.el = el;
11934     /**
11935      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11936      * @type String
11937      */
11938     this.defaultUrl = null;
11939
11940     this.addEvents({
11941         /**
11942          * @event beforeupdate
11943          * Fired before an update is made, return false from your handler and the update is cancelled.
11944          * @param {Roo.Element} el
11945          * @param {String/Object/Function} url
11946          * @param {String/Object} params
11947          */
11948         "beforeupdate": true,
11949         /**
11950          * @event update
11951          * Fired after successful update is made.
11952          * @param {Roo.Element} el
11953          * @param {Object} oResponseObject The response Object
11954          */
11955         "update": true,
11956         /**
11957          * @event failure
11958          * Fired on update failure.
11959          * @param {Roo.Element} el
11960          * @param {Object} oResponseObject The response Object
11961          */
11962         "failure": true
11963     });
11964     var d = Roo.UpdateManager.defaults;
11965     /**
11966      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11967      * @type String
11968      */
11969     this.sslBlankUrl = d.sslBlankUrl;
11970     /**
11971      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11972      * @type Boolean
11973      */
11974     this.disableCaching = d.disableCaching;
11975     /**
11976      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11977      * @type String
11978      */
11979     this.indicatorText = d.indicatorText;
11980     /**
11981      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11982      * @type String
11983      */
11984     this.showLoadIndicator = d.showLoadIndicator;
11985     /**
11986      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11987      * @type Number
11988      */
11989     this.timeout = d.timeout;
11990
11991     /**
11992      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11993      * @type Boolean
11994      */
11995     this.loadScripts = d.loadScripts;
11996
11997     /**
11998      * Transaction object of current executing transaction
11999      */
12000     this.transaction = null;
12001
12002     /**
12003      * @private
12004      */
12005     this.autoRefreshProcId = null;
12006     /**
12007      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.refreshDelegate = this.refresh.createDelegate(this);
12011     /**
12012      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.updateDelegate = this.update.createDelegate(this);
12016     /**
12017      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12018      * @type Function
12019      */
12020     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12021     /**
12022      * @private
12023      */
12024     this.successDelegate = this.processSuccess.createDelegate(this);
12025     /**
12026      * @private
12027      */
12028     this.failureDelegate = this.processFailure.createDelegate(this);
12029
12030     if(!this.renderer){
12031      /**
12032       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12033       */
12034     this.renderer = new Roo.UpdateManager.BasicRenderer();
12035     }
12036     
12037     Roo.UpdateManager.superclass.constructor.call(this);
12038 };
12039
12040 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12041     /**
12042      * Get the Element this UpdateManager is bound to
12043      * @return {Roo.Element} The element
12044      */
12045     getEl : function(){
12046         return this.el;
12047     },
12048     /**
12049      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12050      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
12051 <pre><code>
12052 um.update({<br/>
12053     url: "your-url.php",<br/>
12054     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12055     callback: yourFunction,<br/>
12056     scope: yourObject, //(optional scope)  <br/>
12057     discardUrl: false, <br/>
12058     nocache: false,<br/>
12059     text: "Loading...",<br/>
12060     timeout: 30,<br/>
12061     scripts: false<br/>
12062 });
12063 </code></pre>
12064      * The only required property is url. The optional properties nocache, text and scripts
12065      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12066      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
12067      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12068      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
12069      */
12070     update : function(url, params, callback, discardUrl){
12071         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12072             var method = this.method,
12073                 cfg;
12074             if(typeof url == "object"){ // must be config object
12075                 cfg = url;
12076                 url = cfg.url;
12077                 params = params || cfg.params;
12078                 callback = callback || cfg.callback;
12079                 discardUrl = discardUrl || cfg.discardUrl;
12080                 if(callback && cfg.scope){
12081                     callback = callback.createDelegate(cfg.scope);
12082                 }
12083                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12084                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12085                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12086                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12087                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12088             }
12089             this.showLoading();
12090             if(!discardUrl){
12091                 this.defaultUrl = url;
12092             }
12093             if(typeof url == "function"){
12094                 url = url.call(this);
12095             }
12096
12097             method = method || (params ? "POST" : "GET");
12098             if(method == "GET"){
12099                 url = this.prepareUrl(url);
12100             }
12101
12102             var o = Roo.apply(cfg ||{}, {
12103                 url : url,
12104                 params: params,
12105                 success: this.successDelegate,
12106                 failure: this.failureDelegate,
12107                 callback: undefined,
12108                 timeout: (this.timeout*1000),
12109                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12110             });
12111             Roo.log("updated manager called with timeout of " + o.timeout);
12112             this.transaction = Roo.Ajax.request(o);
12113         }
12114     },
12115
12116     /**
12117      * Performs an async form post, updating this element with the response. If the form has the attribute enctype="multipart/form-data", it assumes it's a file upload.
12118      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12119      * @param {String/HTMLElement} form The form Id or form element
12120      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12121      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12122      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12123      */
12124     formUpdate : function(form, url, reset, callback){
12125         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12126             if(typeof url == "function"){
12127                 url = url.call(this);
12128             }
12129             form = Roo.getDom(form);
12130             this.transaction = Roo.Ajax.request({
12131                 form: form,
12132                 url:url,
12133                 success: this.successDelegate,
12134                 failure: this.failureDelegate,
12135                 timeout: (this.timeout*1000),
12136                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12137             });
12138             this.showLoading.defer(1, this);
12139         }
12140     },
12141
12142     /**
12143      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12144      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12145      */
12146     refresh : function(callback){
12147         if(this.defaultUrl == null){
12148             return;
12149         }
12150         this.update(this.defaultUrl, null, callback, true);
12151     },
12152
12153     /**
12154      * Set this element to auto refresh.
12155      * @param {Number} interval How often to update (in seconds).
12156      * @param {String/Function} url (optional) The url for this request or a function to call to get the url (Defaults to the last used url)
12157      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "&param1=1&param2=2" or as an object {param1: 1, param2: 2}
12158      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12159      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12160      */
12161     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12162         if(refreshNow){
12163             this.update(url || this.defaultUrl, params, callback, true);
12164         }
12165         if(this.autoRefreshProcId){
12166             clearInterval(this.autoRefreshProcId);
12167         }
12168         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12169     },
12170
12171     /**
12172      * Stop auto refresh on this element.
12173      */
12174      stopAutoRefresh : function(){
12175         if(this.autoRefreshProcId){
12176             clearInterval(this.autoRefreshProcId);
12177             delete this.autoRefreshProcId;
12178         }
12179     },
12180
12181     isAutoRefreshing : function(){
12182        return this.autoRefreshProcId ? true : false;
12183     },
12184     /**
12185      * Called to update the element to "Loading" state. Override to perform custom action.
12186      */
12187     showLoading : function(){
12188         if(this.showLoadIndicator){
12189             this.el.update(this.indicatorText);
12190         }
12191     },
12192
12193     /**
12194      * Adds unique parameter to query string if disableCaching = true
12195      * @private
12196      */
12197     prepareUrl : function(url){
12198         if(this.disableCaching){
12199             var append = "_dc=" + (new Date().getTime());
12200             if(url.indexOf("?") !== -1){
12201                 url += "&" + append;
12202             }else{
12203                 url += "?" + append;
12204             }
12205         }
12206         return url;
12207     },
12208
12209     /**
12210      * @private
12211      */
12212     processSuccess : function(response){
12213         this.transaction = null;
12214         if(response.argument.form && response.argument.reset){
12215             try{ // put in try/catch since some older FF releases had problems with this
12216                 response.argument.form.reset();
12217             }catch(e){}
12218         }
12219         if(this.loadScripts){
12220             this.renderer.render(this.el, response, this,
12221                 this.updateComplete.createDelegate(this, [response]));
12222         }else{
12223             this.renderer.render(this.el, response, this);
12224             this.updateComplete(response);
12225         }
12226     },
12227
12228     updateComplete : function(response){
12229         this.fireEvent("update", this.el, response);
12230         if(typeof response.argument.callback == "function"){
12231             response.argument.callback(this.el, true, response);
12232         }
12233     },
12234
12235     /**
12236      * @private
12237      */
12238     processFailure : function(response){
12239         this.transaction = null;
12240         this.fireEvent("failure", this.el, response);
12241         if(typeof response.argument.callback == "function"){
12242             response.argument.callback(this.el, false, response);
12243         }
12244     },
12245
12246     /**
12247      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12248      * @param {Object} renderer The object implementing the render() method
12249      */
12250     setRenderer : function(renderer){
12251         this.renderer = renderer;
12252     },
12253
12254     getRenderer : function(){
12255        return this.renderer;
12256     },
12257
12258     /**
12259      * Set the defaultUrl used for updates
12260      * @param {String/Function} defaultUrl The url or a function to call to get the url
12261      */
12262     setDefaultUrl : function(defaultUrl){
12263         this.defaultUrl = defaultUrl;
12264     },
12265
12266     /**
12267      * Aborts the executing transaction
12268      */
12269     abort : function(){
12270         if(this.transaction){
12271             Roo.Ajax.abort(this.transaction);
12272         }
12273     },
12274
12275     /**
12276      * Returns true if an update is in progress
12277      * @return {Boolean}
12278      */
12279     isUpdating : function(){
12280         if(this.transaction){
12281             return Roo.Ajax.isLoading(this.transaction);
12282         }
12283         return false;
12284     }
12285 });
12286
12287 /**
12288  * @class Roo.UpdateManager.defaults
12289  * @static (not really - but it helps the doc tool)
12290  * The defaults collection enables customizing the default properties of UpdateManager
12291  */
12292    Roo.UpdateManager.defaults = {
12293        /**
12294          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12295          * @type Number
12296          */
12297          timeout : 30,
12298
12299          /**
12300          * True to process scripts by default (Defaults to false).
12301          * @type Boolean
12302          */
12303         loadScripts : false,
12304
12305         /**
12306         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12307         * @type String
12308         */
12309         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12310         /**
12311          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12312          * @type Boolean
12313          */
12314         disableCaching : false,
12315         /**
12316          * Whether to show indicatorText when loading (Defaults to true).
12317          * @type Boolean
12318          */
12319         showLoadIndicator : true,
12320         /**
12321          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12322          * @type String
12323          */
12324         indicatorText : '<div class="loading-indicator">Loading...</div>'
12325    };
12326
12327 /**
12328  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12329  *Usage:
12330  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12331  * @param {String/HTMLElement/Roo.Element} el The element to update
12332  * @param {String} url The url
12333  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12334  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12335  * @static
12336  * @deprecated
12337  * @member Roo.UpdateManager
12338  */
12339 Roo.UpdateManager.updateElement = function(el, url, params, options){
12340     var um = Roo.get(el, true).getUpdateManager();
12341     Roo.apply(um, options);
12342     um.update(url, params, options ? options.callback : null);
12343 };
12344 // alias for backwards compat
12345 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12346 /**
12347  * @class Roo.UpdateManager.BasicRenderer
12348  * Default Content renderer. Updates the elements innerHTML with the responseText.
12349  */
12350 Roo.UpdateManager.BasicRenderer = function(){};
12351
12352 Roo.UpdateManager.BasicRenderer.prototype = {
12353     /**
12354      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12355      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12356      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12357      * @param {Roo.Element} el The element being rendered
12358      * @param {Object} response The YUI Connect response object
12359      * @param {UpdateManager} updateManager The calling update manager
12360      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12361      */
12362      render : function(el, response, updateManager, callback){
12363         el.update(response.responseText, updateManager.loadScripts, callback);
12364     }
12365 };
12366 /*
12367  * Based on:
12368  * Roo JS
12369  * (c)) Alan Knowles
12370  * Licence : LGPL
12371  */
12372
12373
12374 /**
12375  * @class Roo.DomTemplate
12376  * @extends Roo.Template
12377  * An effort at a dom based template engine..
12378  *
12379  * Similar to XTemplate, except it uses dom parsing to create the template..
12380  *
12381  * Supported features:
12382  *
12383  *  Tags:
12384
12385 <pre><code>
12386       {a_variable} - output encoded.
12387       {a_variable.format:("Y-m-d")} - call a method on the variable
12388       {a_variable:raw} - unencoded output
12389       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12390       {a_variable:this.method_on_template(...)} - call a method on the template object.
12391  
12392 </code></pre>
12393  *  The tpl tag:
12394 <pre><code>
12395         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12396         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12397         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12398         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12399   
12400 </code></pre>
12401  *      
12402  */
12403 Roo.DomTemplate = function()
12404 {
12405      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12406      if (this.html) {
12407         this.compile();
12408      }
12409 };
12410
12411
12412 Roo.extend(Roo.DomTemplate, Roo.Template, {
12413     /**
12414      * id counter for sub templates.
12415      */
12416     id : 0,
12417     /**
12418      * flag to indicate if dom parser is inside a pre,
12419      * it will strip whitespace if not.
12420      */
12421     inPre : false,
12422     
12423     /**
12424      * The various sub templates
12425      */
12426     tpls : false,
12427     
12428     
12429     
12430     /**
12431      *
12432      * basic tag replacing syntax
12433      * WORD:WORD()
12434      *
12435      * // you can fake an object call by doing this
12436      *  x.t:(test,tesT) 
12437      * 
12438      */
12439     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12440     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12441     
12442     iterChild : function (node, method) {
12443         
12444         var oldPre = this.inPre;
12445         if (node.tagName == 'PRE') {
12446             this.inPre = true;
12447         }
12448         for( var i = 0; i < node.childNodes.length; i++) {
12449             method.call(this, node.childNodes[i]);
12450         }
12451         this.inPre = oldPre;
12452     },
12453     
12454     
12455     
12456     /**
12457      * compile the template
12458      *
12459      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12460      *
12461      */
12462     compile: function()
12463     {
12464         var s = this.html;
12465         
12466         // covert the html into DOM...
12467         var doc = false;
12468         var div =false;
12469         try {
12470             doc = document.implementation.createHTMLDocument("");
12471             doc.documentElement.innerHTML =   this.html  ;
12472             div = doc.documentElement;
12473         } catch (e) {
12474             // old IE... - nasty -- it causes all sorts of issues.. with
12475             // images getting pulled from server..
12476             div = document.createElement('div');
12477             div.innerHTML = this.html;
12478         }
12479         //doc.documentElement.innerHTML = htmlBody
12480          
12481         
12482         
12483         this.tpls = [];
12484         var _t = this;
12485         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12486         
12487         var tpls = this.tpls;
12488         
12489         // create a top level template from the snippet..
12490         
12491         //Roo.log(div.innerHTML);
12492         
12493         var tpl = {
12494             uid : 'master',
12495             id : this.id++,
12496             attr : false,
12497             value : false,
12498             body : div.innerHTML,
12499             
12500             forCall : false,
12501             execCall : false,
12502             dom : div,
12503             isTop : true
12504             
12505         };
12506         tpls.unshift(tpl);
12507         
12508         
12509         // compile them...
12510         this.tpls = [];
12511         Roo.each(tpls, function(tp){
12512             this.compileTpl(tp);
12513             this.tpls[tp.id] = tp;
12514         }, this);
12515         
12516         this.master = tpls[0];
12517         return this;
12518         
12519         
12520     },
12521     
12522     compileNode : function(node, istop) {
12523         // test for
12524         //Roo.log(node);
12525         
12526         
12527         // skip anything not a tag..
12528         if (node.nodeType != 1) {
12529             if (node.nodeType == 3 && !this.inPre) {
12530                 // reduce white space..
12531                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12532                 
12533             }
12534             return;
12535         }
12536         
12537         var tpl = {
12538             uid : false,
12539             id : false,
12540             attr : false,
12541             value : false,
12542             body : '',
12543             
12544             forCall : false,
12545             execCall : false,
12546             dom : false,
12547             isTop : istop
12548             
12549             
12550         };
12551         
12552         
12553         switch(true) {
12554             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12555             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12556             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12557             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12558             // no default..
12559         }
12560         
12561         
12562         if (!tpl.attr) {
12563             // just itterate children..
12564             this.iterChild(node,this.compileNode);
12565             return;
12566         }
12567         tpl.uid = this.id++;
12568         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12569         node.removeAttribute('roo-'+ tpl.attr);
12570         if (tpl.attr != 'name') {
12571             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12572             node.parentNode.replaceChild(placeholder,  node);
12573         } else {
12574             
12575             var placeholder =  document.createElement('span');
12576             placeholder.className = 'roo-tpl-' + tpl.value;
12577             node.parentNode.replaceChild(placeholder,  node);
12578         }
12579         
12580         // parent now sees '{domtplXXXX}
12581         this.iterChild(node,this.compileNode);
12582         
12583         // we should now have node body...
12584         var div = document.createElement('div');
12585         div.appendChild(node);
12586         tpl.dom = node;
12587         // this has the unfortunate side effect of converting tagged attributes
12588         // eg. href="{...}" into %7C...%7D
12589         // this has been fixed by searching for those combo's although it's a bit hacky..
12590         
12591         
12592         tpl.body = div.innerHTML;
12593         
12594         
12595          
12596         tpl.id = tpl.uid;
12597         switch(tpl.attr) {
12598             case 'for' :
12599                 switch (tpl.value) {
12600                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12601                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12602                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12603                 }
12604                 break;
12605             
12606             case 'exec':
12607                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12608                 break;
12609             
12610             case 'if':     
12611                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12612                 break;
12613             
12614             case 'name':
12615                 tpl.id  = tpl.value; // replace non characters???
12616                 break;
12617             
12618         }
12619         
12620         
12621         this.tpls.push(tpl);
12622         
12623         
12624         
12625     },
12626     
12627     
12628     
12629     
12630     /**
12631      * Compile a segment of the template into a 'sub-template'
12632      *
12633      * 
12634      * 
12635      *
12636      */
12637     compileTpl : function(tpl)
12638     {
12639         var fm = Roo.util.Format;
12640         var useF = this.disableFormats !== true;
12641         
12642         var sep = Roo.isGecko ? "+\n" : ",\n";
12643         
12644         var undef = function(str) {
12645             Roo.debug && Roo.log("Property not found :"  + str);
12646             return '';
12647         };
12648           
12649         //Roo.log(tpl.body);
12650         
12651         
12652         
12653         var fn = function(m, lbrace, name, format, args)
12654         {
12655             //Roo.log("ARGS");
12656             //Roo.log(arguments);
12657             args = args ? args.replace(/\\'/g,"'") : args;
12658             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12659             if (typeof(format) == 'undefined') {
12660                 format =  'htmlEncode'; 
12661             }
12662             if (format == 'raw' ) {
12663                 format = false;
12664             }
12665             
12666             if(name.substr(0, 6) == 'domtpl'){
12667                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12668             }
12669             
12670             // build an array of options to determine if value is undefined..
12671             
12672             // basically get 'xxxx.yyyy' then do
12673             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12674             //    (function () { Roo.log("Property not found"); return ''; })() :
12675             //    ......
12676             
12677             var udef_ar = [];
12678             var lookfor = '';
12679             Roo.each(name.split('.'), function(st) {
12680                 lookfor += (lookfor.length ? '.': '') + st;
12681                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12682             });
12683             
12684             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12685             
12686             
12687             if(format && useF){
12688                 
12689                 args = args ? ',' + args : "";
12690                  
12691                 if(format.substr(0, 5) != "this."){
12692                     format = "fm." + format + '(';
12693                 }else{
12694                     format = 'this.call("'+ format.substr(5) + '", ';
12695                     args = ", values";
12696                 }
12697                 
12698                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12699             }
12700              
12701             if (args && args.length) {
12702                 // called with xxyx.yuu:(test,test)
12703                 // change to ()
12704                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12705             }
12706             // raw.. - :raw modifier..
12707             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12708             
12709         };
12710         var body;
12711         // branched to use + in gecko and [].join() in others
12712         if(Roo.isGecko){
12713             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12714                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12715                     "';};};";
12716         }else{
12717             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12718             body.push(tpl.body.replace(/(\r\n|\n)/g,
12719                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12720             body.push("'].join('');};};");
12721             body = body.join('');
12722         }
12723         
12724         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12725        
12726         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12727         eval(body);
12728         
12729         return this;
12730     },
12731      
12732     /**
12733      * same as applyTemplate, except it's done to one of the subTemplates
12734      * when using named templates, you can do:
12735      *
12736      * var str = pl.applySubTemplate('your-name', values);
12737      *
12738      * 
12739      * @param {Number} id of the template
12740      * @param {Object} values to apply to template
12741      * @param {Object} parent (normaly the instance of this object)
12742      */
12743     applySubTemplate : function(id, values, parent)
12744     {
12745         
12746         
12747         var t = this.tpls[id];
12748         
12749         
12750         try { 
12751             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12752                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12753                 return '';
12754             }
12755         } catch(e) {
12756             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12757             Roo.log(values);
12758           
12759             return '';
12760         }
12761         try { 
12762             
12763             if(t.execCall && t.execCall.call(this, values, parent)){
12764                 return '';
12765             }
12766         } catch(e) {
12767             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12768             Roo.log(values);
12769             return '';
12770         }
12771         
12772         try {
12773             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12774             parent = t.target ? values : parent;
12775             if(t.forCall && vs instanceof Array){
12776                 var buf = [];
12777                 for(var i = 0, len = vs.length; i < len; i++){
12778                     try {
12779                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12780                     } catch (e) {
12781                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12782                         Roo.log(e.body);
12783                         //Roo.log(t.compiled);
12784                         Roo.log(vs[i]);
12785                     }   
12786                 }
12787                 return buf.join('');
12788             }
12789         } catch (e) {
12790             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12791             Roo.log(values);
12792             return '';
12793         }
12794         try {
12795             return t.compiled.call(this, vs, parent);
12796         } catch (e) {
12797             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12798             Roo.log(e.body);
12799             //Roo.log(t.compiled);
12800             Roo.log(values);
12801             return '';
12802         }
12803     },
12804
12805    
12806
12807     applyTemplate : function(values){
12808         return this.master.compiled.call(this, values, {});
12809         //var s = this.subs;
12810     },
12811
12812     apply : function(){
12813         return this.applyTemplate.apply(this, arguments);
12814     }
12815
12816  });
12817
12818 Roo.DomTemplate.from = function(el){
12819     el = Roo.getDom(el);
12820     return new Roo.Domtemplate(el.value || el.innerHTML);
12821 };/*
12822  * Based on:
12823  * Ext JS Library 1.1.1
12824  * Copyright(c) 2006-2007, Ext JS, LLC.
12825  *
12826  * Originally Released Under LGPL - original licence link has changed is not relivant.
12827  *
12828  * Fork - LGPL
12829  * <script type="text/javascript">
12830  */
12831
12832 /**
12833  * @class Roo.util.DelayedTask
12834  * Provides a convenient method of performing setTimeout where a new
12835  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12836  * You can use this class to buffer
12837  * the keypress events for a certain number of milliseconds, and perform only if they stop
12838  * for that amount of time.
12839  * @constructor The parameters to this constructor serve as defaults and are not required.
12840  * @param {Function} fn (optional) The default function to timeout
12841  * @param {Object} scope (optional) The default scope of that timeout
12842  * @param {Array} args (optional) The default Array of arguments
12843  */
12844 Roo.util.DelayedTask = function(fn, scope, args){
12845     var id = null, d, t;
12846
12847     var call = function(){
12848         var now = new Date().getTime();
12849         if(now - t >= d){
12850             clearInterval(id);
12851             id = null;
12852             fn.apply(scope, args || []);
12853         }
12854     };
12855     /**
12856      * Cancels any pending timeout and queues a new one
12857      * @param {Number} delay The milliseconds to delay
12858      * @param {Function} newFn (optional) Overrides function passed to constructor
12859      * @param {Object} newScope (optional) Overrides scope passed to constructor
12860      * @param {Array} newArgs (optional) Overrides args passed to constructor
12861      */
12862     this.delay = function(delay, newFn, newScope, newArgs){
12863         if(id && delay != d){
12864             this.cancel();
12865         }
12866         d = delay;
12867         t = new Date().getTime();
12868         fn = newFn || fn;
12869         scope = newScope || scope;
12870         args = newArgs || args;
12871         if(!id){
12872             id = setInterval(call, d);
12873         }
12874     };
12875
12876     /**
12877      * Cancel the last queued timeout
12878      */
12879     this.cancel = function(){
12880         if(id){
12881             clearInterval(id);
12882             id = null;
12883         }
12884     };
12885 };/*
12886  * Based on:
12887  * Ext JS Library 1.1.1
12888  * Copyright(c) 2006-2007, Ext JS, LLC.
12889  *
12890  * Originally Released Under LGPL - original licence link has changed is not relivant.
12891  *
12892  * Fork - LGPL
12893  * <script type="text/javascript">
12894  */
12895  
12896  
12897 Roo.util.TaskRunner = function(interval){
12898     interval = interval || 10;
12899     var tasks = [], removeQueue = [];
12900     var id = 0;
12901     var running = false;
12902
12903     var stopThread = function(){
12904         running = false;
12905         clearInterval(id);
12906         id = 0;
12907     };
12908
12909     var startThread = function(){
12910         if(!running){
12911             running = true;
12912             id = setInterval(runTasks, interval);
12913         }
12914     };
12915
12916     var removeTask = function(task){
12917         removeQueue.push(task);
12918         if(task.onStop){
12919             task.onStop();
12920         }
12921     };
12922
12923     var runTasks = function(){
12924         if(removeQueue.length > 0){
12925             for(var i = 0, len = removeQueue.length; i < len; i++){
12926                 tasks.remove(removeQueue[i]);
12927             }
12928             removeQueue = [];
12929             if(tasks.length < 1){
12930                 stopThread();
12931                 return;
12932             }
12933         }
12934         var now = new Date().getTime();
12935         for(var i = 0, len = tasks.length; i < len; ++i){
12936             var t = tasks[i];
12937             var itime = now - t.taskRunTime;
12938             if(t.interval <= itime){
12939                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12940                 t.taskRunTime = now;
12941                 if(rt === false || t.taskRunCount === t.repeat){
12942                     removeTask(t);
12943                     return;
12944                 }
12945             }
12946             if(t.duration && t.duration <= (now - t.taskStartTime)){
12947                 removeTask(t);
12948             }
12949         }
12950     };
12951
12952     /**
12953      * Queues a new task.
12954      * @param {Object} task
12955      */
12956     this.start = function(task){
12957         tasks.push(task);
12958         task.taskStartTime = new Date().getTime();
12959         task.taskRunTime = 0;
12960         task.taskRunCount = 0;
12961         startThread();
12962         return task;
12963     };
12964
12965     this.stop = function(task){
12966         removeTask(task);
12967         return task;
12968     };
12969
12970     this.stopAll = function(){
12971         stopThread();
12972         for(var i = 0, len = tasks.length; i < len; i++){
12973             if(tasks[i].onStop){
12974                 tasks[i].onStop();
12975             }
12976         }
12977         tasks = [];
12978         removeQueue = [];
12979     };
12980 };
12981
12982 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12983  * Based on:
12984  * Ext JS Library 1.1.1
12985  * Copyright(c) 2006-2007, Ext JS, LLC.
12986  *
12987  * Originally Released Under LGPL - original licence link has changed is not relivant.
12988  *
12989  * Fork - LGPL
12990  * <script type="text/javascript">
12991  */
12992
12993  
12994 /**
12995  * @class Roo.util.MixedCollection
12996  * @extends Roo.util.Observable
12997  * A Collection class that maintains both numeric indexes and keys and exposes events.
12998  * @constructor
12999  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
13000  * collection (defaults to false)
13001  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
13002  * and return the key value for that item.  This is used when available to look up the key on items that
13003  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
13004  * equivalent to providing an implementation for the {@link #getKey} method.
13005  */
13006 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13007     this.items = [];
13008     this.map = {};
13009     this.keys = [];
13010     this.length = 0;
13011     this.addEvents({
13012         /**
13013          * @event clear
13014          * Fires when the collection is cleared.
13015          */
13016         "clear" : true,
13017         /**
13018          * @event add
13019          * Fires when an item is added to the collection.
13020          * @param {Number} index The index at which the item was added.
13021          * @param {Object} o The item added.
13022          * @param {String} key The key associated with the added item.
13023          */
13024         "add" : true,
13025         /**
13026          * @event replace
13027          * Fires when an item is replaced in the collection.
13028          * @param {String} key he key associated with the new added.
13029          * @param {Object} old The item being replaced.
13030          * @param {Object} new The new item.
13031          */
13032         "replace" : true,
13033         /**
13034          * @event remove
13035          * Fires when an item is removed from the collection.
13036          * @param {Object} o The item being removed.
13037          * @param {String} key (optional) The key associated with the removed item.
13038          */
13039         "remove" : true,
13040         "sort" : true
13041     });
13042     this.allowFunctions = allowFunctions === true;
13043     if(keyFn){
13044         this.getKey = keyFn;
13045     }
13046     Roo.util.MixedCollection.superclass.constructor.call(this);
13047 };
13048
13049 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13050     allowFunctions : false,
13051     
13052 /**
13053  * Adds an item to the collection.
13054  * @param {String} key The key to associate with the item
13055  * @param {Object} o The item to add.
13056  * @return {Object} The item added.
13057  */
13058     add : function(key, o){
13059         if(arguments.length == 1){
13060             o = arguments[0];
13061             key = this.getKey(o);
13062         }
13063         if(typeof key == "undefined" || key === null){
13064             this.length++;
13065             this.items.push(o);
13066             this.keys.push(null);
13067         }else{
13068             var old = this.map[key];
13069             if(old){
13070                 return this.replace(key, o);
13071             }
13072             this.length++;
13073             this.items.push(o);
13074             this.map[key] = o;
13075             this.keys.push(key);
13076         }
13077         this.fireEvent("add", this.length-1, o, key);
13078         return o;
13079     },
13080        
13081 /**
13082   * MixedCollection has a generic way to fetch keys if you implement getKey.
13083 <pre><code>
13084 // normal way
13085 var mc = new Roo.util.MixedCollection();
13086 mc.add(someEl.dom.id, someEl);
13087 mc.add(otherEl.dom.id, otherEl);
13088 //and so on
13089
13090 // using getKey
13091 var mc = new Roo.util.MixedCollection();
13092 mc.getKey = function(el){
13093    return el.dom.id;
13094 };
13095 mc.add(someEl);
13096 mc.add(otherEl);
13097
13098 // or via the constructor
13099 var mc = new Roo.util.MixedCollection(false, function(el){
13100    return el.dom.id;
13101 });
13102 mc.add(someEl);
13103 mc.add(otherEl);
13104 </code></pre>
13105  * @param o {Object} The item for which to find the key.
13106  * @return {Object} The key for the passed item.
13107  */
13108     getKey : function(o){
13109          return o.id; 
13110     },
13111    
13112 /**
13113  * Replaces an item in the collection.
13114  * @param {String} key The key associated with the item to replace, or the item to replace.
13115  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13116  * @return {Object}  The new item.
13117  */
13118     replace : function(key, o){
13119         if(arguments.length == 1){
13120             o = arguments[0];
13121             key = this.getKey(o);
13122         }
13123         var old = this.item(key);
13124         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13125              return this.add(key, o);
13126         }
13127         var index = this.indexOfKey(key);
13128         this.items[index] = o;
13129         this.map[key] = o;
13130         this.fireEvent("replace", key, old, o);
13131         return o;
13132     },
13133    
13134 /**
13135  * Adds all elements of an Array or an Object to the collection.
13136  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13137  * an Array of values, each of which are added to the collection.
13138  */
13139     addAll : function(objs){
13140         if(arguments.length > 1 || objs instanceof Array){
13141             var args = arguments.length > 1 ? arguments : objs;
13142             for(var i = 0, len = args.length; i < len; i++){
13143                 this.add(args[i]);
13144             }
13145         }else{
13146             for(var key in objs){
13147                 if(this.allowFunctions || typeof objs[key] != "function"){
13148                     this.add(key, objs[key]);
13149                 }
13150             }
13151         }
13152     },
13153    
13154 /**
13155  * Executes the specified function once for every item in the collection, passing each
13156  * item as the first and only parameter. returning false from the function will stop the iteration.
13157  * @param {Function} fn The function to execute for each item.
13158  * @param {Object} scope (optional) The scope in which to execute the function.
13159  */
13160     each : function(fn, scope){
13161         var items = [].concat(this.items); // each safe for removal
13162         for(var i = 0, len = items.length; i < len; i++){
13163             if(fn.call(scope || items[i], items[i], i, len) === false){
13164                 break;
13165             }
13166         }
13167     },
13168    
13169 /**
13170  * Executes the specified function once for every key in the collection, passing each
13171  * key, and its associated item as the first two parameters.
13172  * @param {Function} fn The function to execute for each item.
13173  * @param {Object} scope (optional) The scope in which to execute the function.
13174  */
13175     eachKey : function(fn, scope){
13176         for(var i = 0, len = this.keys.length; i < len; i++){
13177             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13178         }
13179     },
13180    
13181 /**
13182  * Returns the first item in the collection which elicits a true return value from the
13183  * passed selection function.
13184  * @param {Function} fn The selection function to execute for each item.
13185  * @param {Object} scope (optional) The scope in which to execute the function.
13186  * @return {Object} The first item in the collection which returned true from the selection function.
13187  */
13188     find : function(fn, scope){
13189         for(var i = 0, len = this.items.length; i < len; i++){
13190             if(fn.call(scope || window, this.items[i], this.keys[i])){
13191                 return this.items[i];
13192             }
13193         }
13194         return null;
13195     },
13196    
13197 /**
13198  * Inserts an item at the specified index in the collection.
13199  * @param {Number} index The index to insert the item at.
13200  * @param {String} key The key to associate with the new item, or the item itself.
13201  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13202  * @return {Object} The item inserted.
13203  */
13204     insert : function(index, key, o){
13205         if(arguments.length == 2){
13206             o = arguments[1];
13207             key = this.getKey(o);
13208         }
13209         if(index >= this.length){
13210             return this.add(key, o);
13211         }
13212         this.length++;
13213         this.items.splice(index, 0, o);
13214         if(typeof key != "undefined" && key != null){
13215             this.map[key] = o;
13216         }
13217         this.keys.splice(index, 0, key);
13218         this.fireEvent("add", index, o, key);
13219         return o;
13220     },
13221    
13222 /**
13223  * Removed an item from the collection.
13224  * @param {Object} o The item to remove.
13225  * @return {Object} The item removed.
13226  */
13227     remove : function(o){
13228         return this.removeAt(this.indexOf(o));
13229     },
13230    
13231 /**
13232  * Remove an item from a specified index in the collection.
13233  * @param {Number} index The index within the collection of the item to remove.
13234  */
13235     removeAt : function(index){
13236         if(index < this.length && index >= 0){
13237             this.length--;
13238             var o = this.items[index];
13239             this.items.splice(index, 1);
13240             var key = this.keys[index];
13241             if(typeof key != "undefined"){
13242                 delete this.map[key];
13243             }
13244             this.keys.splice(index, 1);
13245             this.fireEvent("remove", o, key);
13246         }
13247     },
13248    
13249 /**
13250  * Removed an item associated with the passed key fom the collection.
13251  * @param {String} key The key of the item to remove.
13252  */
13253     removeKey : function(key){
13254         return this.removeAt(this.indexOfKey(key));
13255     },
13256    
13257 /**
13258  * Returns the number of items in the collection.
13259  * @return {Number} the number of items in the collection.
13260  */
13261     getCount : function(){
13262         return this.length; 
13263     },
13264    
13265 /**
13266  * Returns index within the collection of the passed Object.
13267  * @param {Object} o The item to find the index of.
13268  * @return {Number} index of the item.
13269  */
13270     indexOf : function(o){
13271         if(!this.items.indexOf){
13272             for(var i = 0, len = this.items.length; i < len; i++){
13273                 if(this.items[i] == o) {
13274                     return i;
13275                 }
13276             }
13277             return -1;
13278         }else{
13279             return this.items.indexOf(o);
13280         }
13281     },
13282    
13283 /**
13284  * Returns index within the collection of the passed key.
13285  * @param {String} key The key to find the index of.
13286  * @return {Number} index of the key.
13287  */
13288     indexOfKey : function(key){
13289         if(!this.keys.indexOf){
13290             for(var i = 0, len = this.keys.length; i < len; i++){
13291                 if(this.keys[i] == key) {
13292                     return i;
13293                 }
13294             }
13295             return -1;
13296         }else{
13297             return this.keys.indexOf(key);
13298         }
13299     },
13300    
13301 /**
13302  * Returns the item associated with the passed key OR index. Key has priority over index.
13303  * @param {String/Number} key The key or index of the item.
13304  * @return {Object} The item associated with the passed key.
13305  */
13306     item : function(key){
13307         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13308         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13309     },
13310     
13311 /**
13312  * Returns the item at the specified index.
13313  * @param {Number} index The index of the item.
13314  * @return {Object}
13315  */
13316     itemAt : function(index){
13317         return this.items[index];
13318     },
13319     
13320 /**
13321  * Returns the item associated with the passed key.
13322  * @param {String/Number} key The key of the item.
13323  * @return {Object} The item associated with the passed key.
13324  */
13325     key : function(key){
13326         return this.map[key];
13327     },
13328    
13329 /**
13330  * Returns true if the collection contains the passed Object as an item.
13331  * @param {Object} o  The Object to look for in the collection.
13332  * @return {Boolean} True if the collection contains the Object as an item.
13333  */
13334     contains : function(o){
13335         return this.indexOf(o) != -1;
13336     },
13337    
13338 /**
13339  * Returns true if the collection contains the passed Object as a key.
13340  * @param {String} key The key to look for in the collection.
13341  * @return {Boolean} True if the collection contains the Object as a key.
13342  */
13343     containsKey : function(key){
13344         return typeof this.map[key] != "undefined";
13345     },
13346    
13347 /**
13348  * Removes all items from the collection.
13349  */
13350     clear : function(){
13351         this.length = 0;
13352         this.items = [];
13353         this.keys = [];
13354         this.map = {};
13355         this.fireEvent("clear");
13356     },
13357    
13358 /**
13359  * Returns the first item in the collection.
13360  * @return {Object} the first item in the collection..
13361  */
13362     first : function(){
13363         return this.items[0]; 
13364     },
13365    
13366 /**
13367  * Returns the last item in the collection.
13368  * @return {Object} the last item in the collection..
13369  */
13370     last : function(){
13371         return this.items[this.length-1];   
13372     },
13373     
13374     _sort : function(property, dir, fn){
13375         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13376         fn = fn || function(a, b){
13377             return a-b;
13378         };
13379         var c = [], k = this.keys, items = this.items;
13380         for(var i = 0, len = items.length; i < len; i++){
13381             c[c.length] = {key: k[i], value: items[i], index: i};
13382         }
13383         c.sort(function(a, b){
13384             var v = fn(a[property], b[property]) * dsc;
13385             if(v == 0){
13386                 v = (a.index < b.index ? -1 : 1);
13387             }
13388             return v;
13389         });
13390         for(var i = 0, len = c.length; i < len; i++){
13391             items[i] = c[i].value;
13392             k[i] = c[i].key;
13393         }
13394         this.fireEvent("sort", this);
13395     },
13396     
13397     /**
13398      * Sorts this collection with the passed comparison function
13399      * @param {String} direction (optional) "ASC" or "DESC"
13400      * @param {Function} fn (optional) comparison function
13401      */
13402     sort : function(dir, fn){
13403         this._sort("value", dir, fn);
13404     },
13405     
13406     /**
13407      * Sorts this collection by keys
13408      * @param {String} direction (optional) "ASC" or "DESC"
13409      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13410      */
13411     keySort : function(dir, fn){
13412         this._sort("key", dir, fn || function(a, b){
13413             return String(a).toUpperCase()-String(b).toUpperCase();
13414         });
13415     },
13416     
13417     /**
13418      * Returns a range of items in this collection
13419      * @param {Number} startIndex (optional) defaults to 0
13420      * @param {Number} endIndex (optional) default to the last item
13421      * @return {Array} An array of items
13422      */
13423     getRange : function(start, end){
13424         var items = this.items;
13425         if(items.length < 1){
13426             return [];
13427         }
13428         start = start || 0;
13429         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13430         var r = [];
13431         if(start <= end){
13432             for(var i = start; i <= end; i++) {
13433                     r[r.length] = items[i];
13434             }
13435         }else{
13436             for(var i = start; i >= end; i--) {
13437                     r[r.length] = items[i];
13438             }
13439         }
13440         return r;
13441     },
13442         
13443     /**
13444      * Filter the <i>objects</i> in this collection by a specific property. 
13445      * Returns a new collection that has been filtered.
13446      * @param {String} property A property on your objects
13447      * @param {String/RegExp} value Either string that the property values 
13448      * should start with or a RegExp to test against the property
13449      * @return {MixedCollection} The new filtered collection
13450      */
13451     filter : function(property, value){
13452         if(!value.exec){ // not a regex
13453             value = String(value);
13454             if(value.length == 0){
13455                 return this.clone();
13456             }
13457             value = new RegExp("^" + Roo.escapeRe(value), "i");
13458         }
13459         return this.filterBy(function(o){
13460             return o && value.test(o[property]);
13461         });
13462         },
13463     
13464     /**
13465      * Filter by a function. * Returns a new collection that has been filtered.
13466      * The passed function will be called with each 
13467      * object in the collection. If the function returns true, the value is included 
13468      * otherwise it is filtered.
13469      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13470      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13471      * @return {MixedCollection} The new filtered collection
13472      */
13473     filterBy : function(fn, scope){
13474         var r = new Roo.util.MixedCollection();
13475         r.getKey = this.getKey;
13476         var k = this.keys, it = this.items;
13477         for(var i = 0, len = it.length; i < len; i++){
13478             if(fn.call(scope||this, it[i], k[i])){
13479                                 r.add(k[i], it[i]);
13480                         }
13481         }
13482         return r;
13483     },
13484     
13485     /**
13486      * Creates a duplicate of this collection
13487      * @return {MixedCollection}
13488      */
13489     clone : function(){
13490         var r = new Roo.util.MixedCollection();
13491         var k = this.keys, it = this.items;
13492         for(var i = 0, len = it.length; i < len; i++){
13493             r.add(k[i], it[i]);
13494         }
13495         r.getKey = this.getKey;
13496         return r;
13497     }
13498 });
13499 /**
13500  * Returns the item associated with the passed key or index.
13501  * @method
13502  * @param {String/Number} key The key or index of the item.
13503  * @return {Object} The item associated with the passed key.
13504  */
13505 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13506  * Based on:
13507  * Ext JS Library 1.1.1
13508  * Copyright(c) 2006-2007, Ext JS, LLC.
13509  *
13510  * Originally Released Under LGPL - original licence link has changed is not relivant.
13511  *
13512  * Fork - LGPL
13513  * <script type="text/javascript">
13514  */
13515 /**
13516  * @class Roo.util.JSON
13517  * Modified version of Douglas Crockford"s json.js that doesn"t
13518  * mess with the Object prototype 
13519  * http://www.json.org/js.html
13520  * @singleton
13521  */
13522 Roo.util.JSON = new (function(){
13523     var useHasOwn = {}.hasOwnProperty ? true : false;
13524     
13525     // crashes Safari in some instances
13526     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13527     
13528     var pad = function(n) {
13529         return n < 10 ? "0" + n : n;
13530     };
13531     
13532     var m = {
13533         "\b": '\\b',
13534         "\t": '\\t',
13535         "\n": '\\n',
13536         "\f": '\\f',
13537         "\r": '\\r',
13538         '"' : '\\"',
13539         "\\": '\\\\'
13540     };
13541
13542     var encodeString = function(s){
13543         if (/["\\\x00-\x1f]/.test(s)) {
13544             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13545                 var c = m[b];
13546                 if(c){
13547                     return c;
13548                 }
13549                 c = b.charCodeAt();
13550                 return "\\u00" +
13551                     Math.floor(c / 16).toString(16) +
13552                     (c % 16).toString(16);
13553             }) + '"';
13554         }
13555         return '"' + s + '"';
13556     };
13557     
13558     var encodeArray = function(o){
13559         var a = ["["], b, i, l = o.length, v;
13560             for (i = 0; i < l; i += 1) {
13561                 v = o[i];
13562                 switch (typeof v) {
13563                     case "undefined":
13564                     case "function":
13565                     case "unknown":
13566                         break;
13567                     default:
13568                         if (b) {
13569                             a.push(',');
13570                         }
13571                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13572                         b = true;
13573                 }
13574             }
13575             a.push("]");
13576             return a.join("");
13577     };
13578     
13579     var encodeDate = function(o){
13580         return '"' + o.getFullYear() + "-" +
13581                 pad(o.getMonth() + 1) + "-" +
13582                 pad(o.getDate()) + "T" +
13583                 pad(o.getHours()) + ":" +
13584                 pad(o.getMinutes()) + ":" +
13585                 pad(o.getSeconds()) + '"';
13586     };
13587     
13588     /**
13589      * Encodes an Object, Array or other value
13590      * @param {Mixed} o The variable to encode
13591      * @return {String} The JSON string
13592      */
13593     this.encode = function(o)
13594     {
13595         // should this be extended to fully wrap stringify..
13596         
13597         if(typeof o == "undefined" || o === null){
13598             return "null";
13599         }else if(o instanceof Array){
13600             return encodeArray(o);
13601         }else if(o instanceof Date){
13602             return encodeDate(o);
13603         }else if(typeof o == "string"){
13604             return encodeString(o);
13605         }else if(typeof o == "number"){
13606             return isFinite(o) ? String(o) : "null";
13607         }else if(typeof o == "boolean"){
13608             return String(o);
13609         }else {
13610             var a = ["{"], b, i, v;
13611             for (i in o) {
13612                 if(!useHasOwn || o.hasOwnProperty(i)) {
13613                     v = o[i];
13614                     switch (typeof v) {
13615                     case "undefined":
13616                     case "function":
13617                     case "unknown":
13618                         break;
13619                     default:
13620                         if(b){
13621                             a.push(',');
13622                         }
13623                         a.push(this.encode(i), ":",
13624                                 v === null ? "null" : this.encode(v));
13625                         b = true;
13626                     }
13627                 }
13628             }
13629             a.push("}");
13630             return a.join("");
13631         }
13632     };
13633     
13634     /**
13635      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13636      * @param {String} json The JSON string
13637      * @return {Object} The resulting object
13638      */
13639     this.decode = function(json){
13640         
13641         return  /** eval:var:json */ eval("(" + json + ')');
13642     };
13643 })();
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#encode}
13646  * @member Roo encode 
13647  * @method */
13648 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13649 /** 
13650  * Shorthand for {@link Roo.util.JSON#decode}
13651  * @member Roo decode 
13652  * @method */
13653 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13654 /*
13655  * Based on:
13656  * Ext JS Library 1.1.1
13657  * Copyright(c) 2006-2007, Ext JS, LLC.
13658  *
13659  * Originally Released Under LGPL - original licence link has changed is not relivant.
13660  *
13661  * Fork - LGPL
13662  * <script type="text/javascript">
13663  */
13664  
13665 /**
13666  * @class Roo.util.Format
13667  * Reusable data formatting functions
13668  * @singleton
13669  */
13670 Roo.util.Format = function(){
13671     var trimRe = /^\s+|\s+$/g;
13672     return {
13673         /**
13674          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13675          * @param {String} value The string to truncate
13676          * @param {Number} length The maximum length to allow before truncating
13677          * @return {String} The converted text
13678          */
13679         ellipsis : function(value, len){
13680             if(value && value.length > len){
13681                 return value.substr(0, len-3)+"...";
13682             }
13683             return value;
13684         },
13685
13686         /**
13687          * Checks a reference and converts it to empty string if it is undefined
13688          * @param {Mixed} value Reference to check
13689          * @return {Mixed} Empty string if converted, otherwise the original value
13690          */
13691         undef : function(value){
13692             return typeof value != "undefined" ? value : "";
13693         },
13694
13695         /**
13696          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13697          * @param {String} value The string to encode
13698          * @return {String} The encoded text
13699          */
13700         htmlEncode : function(value){
13701             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13702         },
13703
13704         /**
13705          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13706          * @param {String} value The string to decode
13707          * @return {String} The decoded text
13708          */
13709         htmlDecode : function(value){
13710             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13711         },
13712
13713         /**
13714          * Trims any whitespace from either side of a string
13715          * @param {String} value The text to trim
13716          * @return {String} The trimmed text
13717          */
13718         trim : function(value){
13719             return String(value).replace(trimRe, "");
13720         },
13721
13722         /**
13723          * Returns a substring from within an original string
13724          * @param {String} value The original text
13725          * @param {Number} start The start index of the substring
13726          * @param {Number} length The length of the substring
13727          * @return {String} The substring
13728          */
13729         substr : function(value, start, length){
13730             return String(value).substr(start, length);
13731         },
13732
13733         /**
13734          * Converts a string to all lower case letters
13735          * @param {String} value The text to convert
13736          * @return {String} The converted text
13737          */
13738         lowercase : function(value){
13739             return String(value).toLowerCase();
13740         },
13741
13742         /**
13743          * Converts a string to all upper case letters
13744          * @param {String} value The text to convert
13745          * @return {String} The converted text
13746          */
13747         uppercase : function(value){
13748             return String(value).toUpperCase();
13749         },
13750
13751         /**
13752          * Converts the first character only of a string to upper case
13753          * @param {String} value The text to convert
13754          * @return {String} The converted text
13755          */
13756         capitalize : function(value){
13757             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13758         },
13759
13760         // private
13761         call : function(value, fn){
13762             if(arguments.length > 2){
13763                 var args = Array.prototype.slice.call(arguments, 2);
13764                 args.unshift(value);
13765                  
13766                 return /** eval:var:value */  eval(fn).apply(window, args);
13767             }else{
13768                 /** eval:var:value */
13769                 return /** eval:var:value */ eval(fn).call(window, value);
13770             }
13771         },
13772
13773        
13774         /**
13775          * safer version of Math.toFixed..??/
13776          * @param {Number/String} value The numeric value to format
13777          * @param {Number/String} value Decimal places 
13778          * @return {String} The formatted currency string
13779          */
13780         toFixed : function(v, n)
13781         {
13782             // why not use to fixed - precision is buggered???
13783             if (!n) {
13784                 return Math.round(v-0);
13785             }
13786             var fact = Math.pow(10,n+1);
13787             v = (Math.round((v-0)*fact))/fact;
13788             var z = (''+fact).substring(2);
13789             if (v == Math.floor(v)) {
13790                 return Math.floor(v) + '.' + z;
13791             }
13792             
13793             // now just padd decimals..
13794             var ps = String(v).split('.');
13795             var fd = (ps[1] + z);
13796             var r = fd.substring(0,n); 
13797             var rm = fd.substring(n); 
13798             if (rm < 5) {
13799                 return ps[0] + '.' + r;
13800             }
13801             r*=1; // turn it into a number;
13802             r++;
13803             if (String(r).length != n) {
13804                 ps[0]*=1;
13805                 ps[0]++;
13806                 r = String(r).substring(1); // chop the end off.
13807             }
13808             
13809             return ps[0] + '.' + r;
13810              
13811         },
13812         
13813         /**
13814          * Format a number as US currency
13815          * @param {Number/String} value The numeric value to format
13816          * @return {String} The formatted currency string
13817          */
13818         usMoney : function(v){
13819             return '$' + Roo.util.Format.number(v);
13820         },
13821         
13822         /**
13823          * Format a number
13824          * eventually this should probably emulate php's number_format
13825          * @param {Number/String} value The numeric value to format
13826          * @param {Number} decimals number of decimal places
13827          * @param {String} delimiter for thousands (default comma)
13828          * @return {String} The formatted currency string
13829          */
13830         number : function(v, decimals, thousandsDelimiter)
13831         {
13832             // multiply and round.
13833             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13834             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13835             
13836             var mul = Math.pow(10, decimals);
13837             var zero = String(mul).substring(1);
13838             v = (Math.round((v-0)*mul))/mul;
13839             
13840             // if it's '0' number.. then
13841             
13842             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13843             v = String(v);
13844             var ps = v.split('.');
13845             var whole = ps[0];
13846             
13847             var r = /(\d+)(\d{3})/;
13848             // add comma's
13849             
13850             if(thousandsDelimiter.length != 0) {
13851                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13852             } 
13853             
13854             var sub = ps[1] ?
13855                     // has decimals..
13856                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13857                     // does not have decimals
13858                     (decimals ? ('.' + zero) : '');
13859             
13860             
13861             return whole + sub ;
13862         },
13863         
13864         /**
13865          * Parse a value into a formatted date using the specified format pattern.
13866          * @param {Mixed} value The value to format
13867          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13868          * @return {String} The formatted date string
13869          */
13870         date : function(v, format){
13871             if(!v){
13872                 return "";
13873             }
13874             if(!(v instanceof Date)){
13875                 v = new Date(Date.parse(v));
13876             }
13877             return v.dateFormat(format || Roo.util.Format.defaults.date);
13878         },
13879
13880         /**
13881          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13882          * @param {String} format Any valid date format string
13883          * @return {Function} The date formatting function
13884          */
13885         dateRenderer : function(format){
13886             return function(v){
13887                 return Roo.util.Format.date(v, format);  
13888             };
13889         },
13890
13891         // private
13892         stripTagsRE : /<\/?[^>]+>/gi,
13893         
13894         /**
13895          * Strips all HTML tags
13896          * @param {Mixed} value The text from which to strip tags
13897          * @return {String} The stripped text
13898          */
13899         stripTags : function(v){
13900             return !v ? v : String(v).replace(this.stripTagsRE, "");
13901         }
13902     };
13903 }();
13904 Roo.util.Format.defaults = {
13905     date : 'd/M/Y'
13906 };/*
13907  * Based on:
13908  * Ext JS Library 1.1.1
13909  * Copyright(c) 2006-2007, Ext JS, LLC.
13910  *
13911  * Originally Released Under LGPL - original licence link has changed is not relivant.
13912  *
13913  * Fork - LGPL
13914  * <script type="text/javascript">
13915  */
13916
13917
13918  
13919
13920 /**
13921  * @class Roo.MasterTemplate
13922  * @extends Roo.Template
13923  * Provides a template that can have child templates. The syntax is:
13924 <pre><code>
13925 var t = new Roo.MasterTemplate(
13926         '&lt;select name="{name}"&gt;',
13927                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13928         '&lt;/select&gt;'
13929 );
13930 t.add('options', {value: 'foo', text: 'bar'});
13931 // or you can add multiple child elements in one shot
13932 t.addAll('options', [
13933     {value: 'foo', text: 'bar'},
13934     {value: 'foo2', text: 'bar2'},
13935     {value: 'foo3', text: 'bar3'}
13936 ]);
13937 // then append, applying the master template values
13938 t.append('my-form', {name: 'my-select'});
13939 </code></pre>
13940 * A name attribute for the child template is not required if you have only one child
13941 * template or you want to refer to them by index.
13942  */
13943 Roo.MasterTemplate = function(){
13944     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13945     this.originalHtml = this.html;
13946     var st = {};
13947     var m, re = this.subTemplateRe;
13948     re.lastIndex = 0;
13949     var subIndex = 0;
13950     while(m = re.exec(this.html)){
13951         var name = m[1], content = m[2];
13952         st[subIndex] = {
13953             name: name,
13954             index: subIndex,
13955             buffer: [],
13956             tpl : new Roo.Template(content)
13957         };
13958         if(name){
13959             st[name] = st[subIndex];
13960         }
13961         st[subIndex].tpl.compile();
13962         st[subIndex].tpl.call = this.call.createDelegate(this);
13963         subIndex++;
13964     }
13965     this.subCount = subIndex;
13966     this.subs = st;
13967 };
13968 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13969     /**
13970     * The regular expression used to match sub templates
13971     * @type RegExp
13972     * @property
13973     */
13974     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13975
13976     /**
13977      * Applies the passed values to a child template.
13978      * @param {String/Number} name (optional) The name or index of the child template
13979      * @param {Array/Object} values The values to be applied to the template
13980      * @return {MasterTemplate} this
13981      */
13982      add : function(name, values){
13983         if(arguments.length == 1){
13984             values = arguments[0];
13985             name = 0;
13986         }
13987         var s = this.subs[name];
13988         s.buffer[s.buffer.length] = s.tpl.apply(values);
13989         return this;
13990     },
13991
13992     /**
13993      * Applies all the passed values to a child template.
13994      * @param {String/Number} name (optional) The name or index of the child template
13995      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13996      * @param {Boolean} reset (optional) True to reset the template first
13997      * @return {MasterTemplate} this
13998      */
13999     fill : function(name, values, reset){
14000         var a = arguments;
14001         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
14002             values = a[0];
14003             name = 0;
14004             reset = a[1];
14005         }
14006         if(reset){
14007             this.reset();
14008         }
14009         for(var i = 0, len = values.length; i < len; i++){
14010             this.add(name, values[i]);
14011         }
14012         return this;
14013     },
14014
14015     /**
14016      * Resets the template for reuse
14017      * @return {MasterTemplate} this
14018      */
14019      reset : function(){
14020         var s = this.subs;
14021         for(var i = 0; i < this.subCount; i++){
14022             s[i].buffer = [];
14023         }
14024         return this;
14025     },
14026
14027     applyTemplate : function(values){
14028         var s = this.subs;
14029         var replaceIndex = -1;
14030         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14031             return s[++replaceIndex].buffer.join("");
14032         });
14033         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14034     },
14035
14036     apply : function(){
14037         return this.applyTemplate.apply(this, arguments);
14038     },
14039
14040     compile : function(){return this;}
14041 });
14042
14043 /**
14044  * Alias for fill().
14045  * @method
14046  */
14047 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14048  /**
14049  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14050  * var tpl = Roo.MasterTemplate.from('element-id');
14051  * @param {String/HTMLElement} el
14052  * @param {Object} config
14053  * @static
14054  */
14055 Roo.MasterTemplate.from = function(el, config){
14056     el = Roo.getDom(el);
14057     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14058 };/*
14059  * Based on:
14060  * Ext JS Library 1.1.1
14061  * Copyright(c) 2006-2007, Ext JS, LLC.
14062  *
14063  * Originally Released Under LGPL - original licence link has changed is not relivant.
14064  *
14065  * Fork - LGPL
14066  * <script type="text/javascript">
14067  */
14068
14069  
14070 /**
14071  * @class Roo.util.CSS
14072  * Utility class for manipulating CSS rules
14073  * @singleton
14074  */
14075 Roo.util.CSS = function(){
14076         var rules = null;
14077         var doc = document;
14078
14079     var camelRe = /(-[a-z])/gi;
14080     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14081
14082    return {
14083    /**
14084     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14085     * tag and appended to the HEAD of the document.
14086     * @param {String|Object} cssText The text containing the css rules
14087     * @param {String} id An id to add to the stylesheet for later removal
14088     * @return {StyleSheet}
14089     */
14090     createStyleSheet : function(cssText, id){
14091         var ss;
14092         var head = doc.getElementsByTagName("head")[0];
14093         var nrules = doc.createElement("style");
14094         nrules.setAttribute("type", "text/css");
14095         if(id){
14096             nrules.setAttribute("id", id);
14097         }
14098         if (typeof(cssText) != 'string') {
14099             // support object maps..
14100             // not sure if this a good idea.. 
14101             // perhaps it should be merged with the general css handling
14102             // and handle js style props.
14103             var cssTextNew = [];
14104             for(var n in cssText) {
14105                 var citems = [];
14106                 for(var k in cssText[n]) {
14107                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14108                 }
14109                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14110                 
14111             }
14112             cssText = cssTextNew.join("\n");
14113             
14114         }
14115        
14116        
14117        if(Roo.isIE){
14118            head.appendChild(nrules);
14119            ss = nrules.styleSheet;
14120            ss.cssText = cssText;
14121        }else{
14122            try{
14123                 nrules.appendChild(doc.createTextNode(cssText));
14124            }catch(e){
14125                nrules.cssText = cssText; 
14126            }
14127            head.appendChild(nrules);
14128            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14129        }
14130        this.cacheStyleSheet(ss);
14131        return ss;
14132    },
14133
14134    /**
14135     * Removes a style or link tag by id
14136     * @param {String} id The id of the tag
14137     */
14138    removeStyleSheet : function(id){
14139        var existing = doc.getElementById(id);
14140        if(existing){
14141            existing.parentNode.removeChild(existing);
14142        }
14143    },
14144
14145    /**
14146     * Dynamically swaps an existing stylesheet reference for a new one
14147     * @param {String} id The id of an existing link tag to remove
14148     * @param {String} url The href of the new stylesheet to include
14149     */
14150    swapStyleSheet : function(id, url){
14151        this.removeStyleSheet(id);
14152        var ss = doc.createElement("link");
14153        ss.setAttribute("rel", "stylesheet");
14154        ss.setAttribute("type", "text/css");
14155        ss.setAttribute("id", id);
14156        ss.setAttribute("href", url);
14157        doc.getElementsByTagName("head")[0].appendChild(ss);
14158    },
14159    
14160    /**
14161     * Refresh the rule cache if you have dynamically added stylesheets
14162     * @return {Object} An object (hash) of rules indexed by selector
14163     */
14164    refreshCache : function(){
14165        return this.getRules(true);
14166    },
14167
14168    // private
14169    cacheStyleSheet : function(stylesheet){
14170        if(!rules){
14171            rules = {};
14172        }
14173        try{// try catch for cross domain access issue
14174            var ssRules = stylesheet.cssRules || stylesheet.rules;
14175            for(var j = ssRules.length-1; j >= 0; --j){
14176                rules[ssRules[j].selectorText] = ssRules[j];
14177            }
14178        }catch(e){}
14179    },
14180    
14181    /**
14182     * Gets all css rules for the document
14183     * @param {Boolean} refreshCache true to refresh the internal cache
14184     * @return {Object} An object (hash) of rules indexed by selector
14185     */
14186    getRules : function(refreshCache){
14187                 if(rules == null || refreshCache){
14188                         rules = {};
14189                         var ds = doc.styleSheets;
14190                         for(var i =0, len = ds.length; i < len; i++){
14191                             try{
14192                         this.cacheStyleSheet(ds[i]);
14193                     }catch(e){} 
14194                 }
14195                 }
14196                 return rules;
14197         },
14198         
14199         /**
14200     * Gets an an individual CSS rule by selector(s)
14201     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14202     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14203     * @return {CSSRule} The CSS rule or null if one is not found
14204     */
14205    getRule : function(selector, refreshCache){
14206                 var rs = this.getRules(refreshCache);
14207                 if(!(selector instanceof Array)){
14208                     return rs[selector];
14209                 }
14210                 for(var i = 0; i < selector.length; i++){
14211                         if(rs[selector[i]]){
14212                                 return rs[selector[i]];
14213                         }
14214                 }
14215                 return null;
14216         },
14217         
14218         
14219         /**
14220     * Updates a rule property
14221     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14222     * @param {String} property The css property
14223     * @param {String} value The new value for the property
14224     * @return {Boolean} true If a rule was found and updated
14225     */
14226    updateRule : function(selector, property, value){
14227                 if(!(selector instanceof Array)){
14228                         var rule = this.getRule(selector);
14229                         if(rule){
14230                                 rule.style[property.replace(camelRe, camelFn)] = value;
14231                                 return true;
14232                         }
14233                 }else{
14234                         for(var i = 0; i < selector.length; i++){
14235                                 if(this.updateRule(selector[i], property, value)){
14236                                         return true;
14237                                 }
14238                         }
14239                 }
14240                 return false;
14241         }
14242    };   
14243 }();/*
14244  * Based on:
14245  * Ext JS Library 1.1.1
14246  * Copyright(c) 2006-2007, Ext JS, LLC.
14247  *
14248  * Originally Released Under LGPL - original licence link has changed is not relivant.
14249  *
14250  * Fork - LGPL
14251  * <script type="text/javascript">
14252  */
14253
14254  
14255
14256 /**
14257  * @class Roo.util.ClickRepeater
14258  * @extends Roo.util.Observable
14259  * 
14260  * A wrapper class which can be applied to any element. Fires a "click" event while the
14261  * mouse is pressed. The interval between firings may be specified in the config but
14262  * defaults to 10 milliseconds.
14263  * 
14264  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14265  * 
14266  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14267  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14268  * Similar to an autorepeat key delay.
14269  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14270  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14271  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14272  *           "interval" and "delay" are ignored. "immediate" is honored.
14273  * @cfg {Boolean} preventDefault True to prevent the default click event
14274  * @cfg {Boolean} stopDefault True to stop the default click event
14275  * 
14276  * @history
14277  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14278  *     2007-02-02 jvs Renamed to ClickRepeater
14279  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14280  *
14281  *  @constructor
14282  * @param {String/HTMLElement/Element} el The element to listen on
14283  * @param {Object} config
14284  **/
14285 Roo.util.ClickRepeater = function(el, config)
14286 {
14287     this.el = Roo.get(el);
14288     this.el.unselectable();
14289
14290     Roo.apply(this, config);
14291
14292     this.addEvents({
14293     /**
14294      * @event mousedown
14295      * Fires when the mouse button is depressed.
14296      * @param {Roo.util.ClickRepeater} this
14297      */
14298         "mousedown" : true,
14299     /**
14300      * @event click
14301      * Fires on a specified interval during the time the element is pressed.
14302      * @param {Roo.util.ClickRepeater} this
14303      */
14304         "click" : true,
14305     /**
14306      * @event mouseup
14307      * Fires when the mouse key is released.
14308      * @param {Roo.util.ClickRepeater} this
14309      */
14310         "mouseup" : true
14311     });
14312
14313     this.el.on("mousedown", this.handleMouseDown, this);
14314     if(this.preventDefault || this.stopDefault){
14315         this.el.on("click", function(e){
14316             if(this.preventDefault){
14317                 e.preventDefault();
14318             }
14319             if(this.stopDefault){
14320                 e.stopEvent();
14321             }
14322         }, this);
14323     }
14324
14325     // allow inline handler
14326     if(this.handler){
14327         this.on("click", this.handler,  this.scope || this);
14328     }
14329
14330     Roo.util.ClickRepeater.superclass.constructor.call(this);
14331 };
14332
14333 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14334     interval : 20,
14335     delay: 250,
14336     preventDefault : true,
14337     stopDefault : false,
14338     timer : 0,
14339
14340     // private
14341     handleMouseDown : function(){
14342         clearTimeout(this.timer);
14343         this.el.blur();
14344         if(this.pressClass){
14345             this.el.addClass(this.pressClass);
14346         }
14347         this.mousedownTime = new Date();
14348
14349         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14350         this.el.on("mouseout", this.handleMouseOut, this);
14351
14352         this.fireEvent("mousedown", this);
14353         this.fireEvent("click", this);
14354         
14355         this.timer = this.click.defer(this.delay || this.interval, this);
14356     },
14357
14358     // private
14359     click : function(){
14360         this.fireEvent("click", this);
14361         this.timer = this.click.defer(this.getInterval(), this);
14362     },
14363
14364     // private
14365     getInterval: function(){
14366         if(!this.accelerate){
14367             return this.interval;
14368         }
14369         var pressTime = this.mousedownTime.getElapsed();
14370         if(pressTime < 500){
14371             return 400;
14372         }else if(pressTime < 1700){
14373             return 320;
14374         }else if(pressTime < 2600){
14375             return 250;
14376         }else if(pressTime < 3500){
14377             return 180;
14378         }else if(pressTime < 4400){
14379             return 140;
14380         }else if(pressTime < 5300){
14381             return 80;
14382         }else if(pressTime < 6200){
14383             return 50;
14384         }else{
14385             return 10;
14386         }
14387     },
14388
14389     // private
14390     handleMouseOut : function(){
14391         clearTimeout(this.timer);
14392         if(this.pressClass){
14393             this.el.removeClass(this.pressClass);
14394         }
14395         this.el.on("mouseover", this.handleMouseReturn, this);
14396     },
14397
14398     // private
14399     handleMouseReturn : function(){
14400         this.el.un("mouseover", this.handleMouseReturn);
14401         if(this.pressClass){
14402             this.el.addClass(this.pressClass);
14403         }
14404         this.click();
14405     },
14406
14407     // private
14408     handleMouseUp : function(){
14409         clearTimeout(this.timer);
14410         this.el.un("mouseover", this.handleMouseReturn);
14411         this.el.un("mouseout", this.handleMouseOut);
14412         Roo.get(document).un("mouseup", this.handleMouseUp);
14413         this.el.removeClass(this.pressClass);
14414         this.fireEvent("mouseup", this);
14415     }
14416 });/*
14417  * Based on:
14418  * Ext JS Library 1.1.1
14419  * Copyright(c) 2006-2007, Ext JS, LLC.
14420  *
14421  * Originally Released Under LGPL - original licence link has changed is not relivant.
14422  *
14423  * Fork - LGPL
14424  * <script type="text/javascript">
14425  */
14426
14427  
14428 /**
14429  * @class Roo.KeyNav
14430  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14431  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14432  * way to implement custom navigation schemes for any UI component.</p>
14433  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14434  * pageUp, pageDown, del, home, end.  Usage:</p>
14435  <pre><code>
14436 var nav = new Roo.KeyNav("my-element", {
14437     "left" : function(e){
14438         this.moveLeft(e.ctrlKey);
14439     },
14440     "right" : function(e){
14441         this.moveRight(e.ctrlKey);
14442     },
14443     "enter" : function(e){
14444         this.save();
14445     },
14446     scope : this
14447 });
14448 </code></pre>
14449  * @constructor
14450  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14451  * @param {Object} config The config
14452  */
14453 Roo.KeyNav = function(el, config){
14454     this.el = Roo.get(el);
14455     Roo.apply(this, config);
14456     if(!this.disabled){
14457         this.disabled = true;
14458         this.enable();
14459     }
14460 };
14461
14462 Roo.KeyNav.prototype = {
14463     /**
14464      * @cfg {Boolean} disabled
14465      * True to disable this KeyNav instance (defaults to false)
14466      */
14467     disabled : false,
14468     /**
14469      * @cfg {String} defaultEventAction
14470      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14471      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14472      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14473      */
14474     defaultEventAction: "stopEvent",
14475     /**
14476      * @cfg {Boolean} forceKeyDown
14477      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14478      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14479      * handle keydown instead of keypress.
14480      */
14481     forceKeyDown : false,
14482
14483     // private
14484     prepareEvent : function(e){
14485         var k = e.getKey();
14486         var h = this.keyToHandler[k];
14487         //if(h && this[h]){
14488         //    e.stopPropagation();
14489         //}
14490         if(Roo.isSafari && h && k >= 37 && k <= 40){
14491             e.stopEvent();
14492         }
14493     },
14494
14495     // private
14496     relay : function(e){
14497         var k = e.getKey();
14498         var h = this.keyToHandler[k];
14499         if(h && this[h]){
14500             if(this.doRelay(e, this[h], h) !== true){
14501                 e[this.defaultEventAction]();
14502             }
14503         }
14504     },
14505
14506     // private
14507     doRelay : function(e, h, hname){
14508         return h.call(this.scope || this, e);
14509     },
14510
14511     // possible handlers
14512     enter : false,
14513     left : false,
14514     right : false,
14515     up : false,
14516     down : false,
14517     tab : false,
14518     esc : false,
14519     pageUp : false,
14520     pageDown : false,
14521     del : false,
14522     home : false,
14523     end : false,
14524
14525     // quick lookup hash
14526     keyToHandler : {
14527         37 : "left",
14528         39 : "right",
14529         38 : "up",
14530         40 : "down",
14531         33 : "pageUp",
14532         34 : "pageDown",
14533         46 : "del",
14534         36 : "home",
14535         35 : "end",
14536         13 : "enter",
14537         27 : "esc",
14538         9  : "tab"
14539     },
14540
14541         /**
14542          * Enable this KeyNav
14543          */
14544         enable: function(){
14545                 if(this.disabled){
14546             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14547             // the EventObject will normalize Safari automatically
14548             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14549                 this.el.on("keydown", this.relay,  this);
14550             }else{
14551                 this.el.on("keydown", this.prepareEvent,  this);
14552                 this.el.on("keypress", this.relay,  this);
14553             }
14554                     this.disabled = false;
14555                 }
14556         },
14557
14558         /**
14559          * Disable this KeyNav
14560          */
14561         disable: function(){
14562                 if(!this.disabled){
14563                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14564                 this.el.un("keydown", this.relay);
14565             }else{
14566                 this.el.un("keydown", this.prepareEvent);
14567                 this.el.un("keypress", this.relay);
14568             }
14569                     this.disabled = true;
14570                 }
14571         }
14572 };/*
14573  * Based on:
14574  * Ext JS Library 1.1.1
14575  * Copyright(c) 2006-2007, Ext JS, LLC.
14576  *
14577  * Originally Released Under LGPL - original licence link has changed is not relivant.
14578  *
14579  * Fork - LGPL
14580  * <script type="text/javascript">
14581  */
14582
14583  
14584 /**
14585  * @class Roo.KeyMap
14586  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14587  * The constructor accepts the same config object as defined by {@link #addBinding}.
14588  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14589  * combination it will call the function with this signature (if the match is a multi-key
14590  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14591  * A KeyMap can also handle a string representation of keys.<br />
14592  * Usage:
14593  <pre><code>
14594 // map one key by key code
14595 var map = new Roo.KeyMap("my-element", {
14596     key: 13, // or Roo.EventObject.ENTER
14597     fn: myHandler,
14598     scope: myObject
14599 });
14600
14601 // map multiple keys to one action by string
14602 var map = new Roo.KeyMap("my-element", {
14603     key: "a\r\n\t",
14604     fn: myHandler,
14605     scope: myObject
14606 });
14607
14608 // map multiple keys to multiple actions by strings and array of codes
14609 var map = new Roo.KeyMap("my-element", [
14610     {
14611         key: [10,13],
14612         fn: function(){ alert("Return was pressed"); }
14613     }, {
14614         key: "abc",
14615         fn: function(){ alert('a, b or c was pressed'); }
14616     }, {
14617         key: "\t",
14618         ctrl:true,
14619         shift:true,
14620         fn: function(){ alert('Control + shift + tab was pressed.'); }
14621     }
14622 ]);
14623 </code></pre>
14624  * <b>Note: A KeyMap starts enabled</b>
14625  * @constructor
14626  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14627  * @param {Object} config The config (see {@link #addBinding})
14628  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14629  */
14630 Roo.KeyMap = function(el, config, eventName){
14631     this.el  = Roo.get(el);
14632     this.eventName = eventName || "keydown";
14633     this.bindings = [];
14634     if(config){
14635         this.addBinding(config);
14636     }
14637     this.enable();
14638 };
14639
14640 Roo.KeyMap.prototype = {
14641     /**
14642      * True to stop the event from bubbling and prevent the default browser action if the
14643      * key was handled by the KeyMap (defaults to false)
14644      * @type Boolean
14645      */
14646     stopEvent : false,
14647
14648     /**
14649      * Add a new binding to this KeyMap. The following config object properties are supported:
14650      * <pre>
14651 Property    Type             Description
14652 ----------  ---------------  ----------------------------------------------------------------------
14653 key         String/Array     A single keycode or an array of keycodes to handle
14654 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14655 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14656 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14657 fn          Function         The function to call when KeyMap finds the expected key combination
14658 scope       Object           The scope of the callback function
14659 </pre>
14660      *
14661      * Usage:
14662      * <pre><code>
14663 // Create a KeyMap
14664 var map = new Roo.KeyMap(document, {
14665     key: Roo.EventObject.ENTER,
14666     fn: handleKey,
14667     scope: this
14668 });
14669
14670 //Add a new binding to the existing KeyMap later
14671 map.addBinding({
14672     key: 'abc',
14673     shift: true,
14674     fn: handleKey,
14675     scope: this
14676 });
14677 </code></pre>
14678      * @param {Object/Array} config A single KeyMap config or an array of configs
14679      */
14680         addBinding : function(config){
14681         if(config instanceof Array){
14682             for(var i = 0, len = config.length; i < len; i++){
14683                 this.addBinding(config[i]);
14684             }
14685             return;
14686         }
14687         var keyCode = config.key,
14688             shift = config.shift, 
14689             ctrl = config.ctrl, 
14690             alt = config.alt,
14691             fn = config.fn,
14692             scope = config.scope;
14693         if(typeof keyCode == "string"){
14694             var ks = [];
14695             var keyString = keyCode.toUpperCase();
14696             for(var j = 0, len = keyString.length; j < len; j++){
14697                 ks.push(keyString.charCodeAt(j));
14698             }
14699             keyCode = ks;
14700         }
14701         var keyArray = keyCode instanceof Array;
14702         var handler = function(e){
14703             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14704                 var k = e.getKey();
14705                 if(keyArray){
14706                     for(var i = 0, len = keyCode.length; i < len; i++){
14707                         if(keyCode[i] == k){
14708                           if(this.stopEvent){
14709                               e.stopEvent();
14710                           }
14711                           fn.call(scope || window, k, e);
14712                           return;
14713                         }
14714                     }
14715                 }else{
14716                     if(k == keyCode){
14717                         if(this.stopEvent){
14718                            e.stopEvent();
14719                         }
14720                         fn.call(scope || window, k, e);
14721                     }
14722                 }
14723             }
14724         };
14725         this.bindings.push(handler);  
14726         },
14727
14728     /**
14729      * Shorthand for adding a single key listener
14730      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14731      * following options:
14732      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14733      * @param {Function} fn The function to call
14734      * @param {Object} scope (optional) The scope of the function
14735      */
14736     on : function(key, fn, scope){
14737         var keyCode, shift, ctrl, alt;
14738         if(typeof key == "object" && !(key instanceof Array)){
14739             keyCode = key.key;
14740             shift = key.shift;
14741             ctrl = key.ctrl;
14742             alt = key.alt;
14743         }else{
14744             keyCode = key;
14745         }
14746         this.addBinding({
14747             key: keyCode,
14748             shift: shift,
14749             ctrl: ctrl,
14750             alt: alt,
14751             fn: fn,
14752             scope: scope
14753         })
14754     },
14755
14756     // private
14757     handleKeyDown : function(e){
14758             if(this.enabled){ //just in case
14759             var b = this.bindings;
14760             for(var i = 0, len = b.length; i < len; i++){
14761                 b[i].call(this, e);
14762             }
14763             }
14764         },
14765         
14766         /**
14767          * Returns true if this KeyMap is enabled
14768          * @return {Boolean} 
14769          */
14770         isEnabled : function(){
14771             return this.enabled;  
14772         },
14773         
14774         /**
14775          * Enables this KeyMap
14776          */
14777         enable: function(){
14778                 if(!this.enabled){
14779                     this.el.on(this.eventName, this.handleKeyDown, this);
14780                     this.enabled = true;
14781                 }
14782         },
14783
14784         /**
14785          * Disable this KeyMap
14786          */
14787         disable: function(){
14788                 if(this.enabled){
14789                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14790                     this.enabled = false;
14791                 }
14792         }
14793 };/*
14794  * Based on:
14795  * Ext JS Library 1.1.1
14796  * Copyright(c) 2006-2007, Ext JS, LLC.
14797  *
14798  * Originally Released Under LGPL - original licence link has changed is not relivant.
14799  *
14800  * Fork - LGPL
14801  * <script type="text/javascript">
14802  */
14803
14804  
14805 /**
14806  * @class Roo.util.TextMetrics
14807  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14808  * wide, in pixels, a given block of text will be.
14809  * @singleton
14810  */
14811 Roo.util.TextMetrics = function(){
14812     var shared;
14813     return {
14814         /**
14815          * Measures the size of the specified text
14816          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14817          * that can affect the size of the rendered text
14818          * @param {String} text The text to measure
14819          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14820          * in order to accurately measure the text height
14821          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14822          */
14823         measure : function(el, text, fixedWidth){
14824             if(!shared){
14825                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14826             }
14827             shared.bind(el);
14828             shared.setFixedWidth(fixedWidth || 'auto');
14829             return shared.getSize(text);
14830         },
14831
14832         /**
14833          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14834          * the overhead of multiple calls to initialize the style properties on each measurement.
14835          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14836          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14837          * in order to accurately measure the text height
14838          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14839          */
14840         createInstance : function(el, fixedWidth){
14841             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14842         }
14843     };
14844 }();
14845
14846  
14847
14848 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14849     var ml = new Roo.Element(document.createElement('div'));
14850     document.body.appendChild(ml.dom);
14851     ml.position('absolute');
14852     ml.setLeftTop(-1000, -1000);
14853     ml.hide();
14854
14855     if(fixedWidth){
14856         ml.setWidth(fixedWidth);
14857     }
14858      
14859     var instance = {
14860         /**
14861          * Returns the size of the specified text based on the internal element's style and width properties
14862          * @memberOf Roo.util.TextMetrics.Instance#
14863          * @param {String} text The text to measure
14864          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14865          */
14866         getSize : function(text){
14867             ml.update(text);
14868             var s = ml.getSize();
14869             ml.update('');
14870             return s;
14871         },
14872
14873         /**
14874          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14875          * that can affect the size of the rendered text
14876          * @memberOf Roo.util.TextMetrics.Instance#
14877          * @param {String/HTMLElement} el The element, dom node or id
14878          */
14879         bind : function(el){
14880             ml.setStyle(
14881                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14882             );
14883         },
14884
14885         /**
14886          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14887          * to set a fixed width in order to accurately measure the text height.
14888          * @memberOf Roo.util.TextMetrics.Instance#
14889          * @param {Number} width The width to set on the element
14890          */
14891         setFixedWidth : function(width){
14892             ml.setWidth(width);
14893         },
14894
14895         /**
14896          * Returns the measured width of the specified text
14897          * @memberOf Roo.util.TextMetrics.Instance#
14898          * @param {String} text The text to measure
14899          * @return {Number} width The width in pixels
14900          */
14901         getWidth : function(text){
14902             ml.dom.style.width = 'auto';
14903             return this.getSize(text).width;
14904         },
14905
14906         /**
14907          * Returns the measured height of the specified text.  For multiline text, be sure to call
14908          * {@link #setFixedWidth} if necessary.
14909          * @memberOf Roo.util.TextMetrics.Instance#
14910          * @param {String} text The text to measure
14911          * @return {Number} height The height in pixels
14912          */
14913         getHeight : function(text){
14914             return this.getSize(text).height;
14915         }
14916     };
14917
14918     instance.bind(bindTo);
14919
14920     return instance;
14921 };
14922
14923 // backwards compat
14924 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14925  * Based on:
14926  * Ext JS Library 1.1.1
14927  * Copyright(c) 2006-2007, Ext JS, LLC.
14928  *
14929  * Originally Released Under LGPL - original licence link has changed is not relivant.
14930  *
14931  * Fork - LGPL
14932  * <script type="text/javascript">
14933  */
14934
14935 /**
14936  * @class Roo.state.Provider
14937  * Abstract base class for state provider implementations. This class provides methods
14938  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14939  * Provider interface.
14940  */
14941 Roo.state.Provider = function(){
14942     /**
14943      * @event statechange
14944      * Fires when a state change occurs.
14945      * @param {Provider} this This state provider
14946      * @param {String} key The state key which was changed
14947      * @param {String} value The encoded value for the state
14948      */
14949     this.addEvents({
14950         "statechange": true
14951     });
14952     this.state = {};
14953     Roo.state.Provider.superclass.constructor.call(this);
14954 };
14955 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14956     /**
14957      * Returns the current value for a key
14958      * @param {String} name The key name
14959      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14960      * @return {Mixed} The state data
14961      */
14962     get : function(name, defaultValue){
14963         return typeof this.state[name] == "undefined" ?
14964             defaultValue : this.state[name];
14965     },
14966     
14967     /**
14968      * Clears a value from the state
14969      * @param {String} name The key name
14970      */
14971     clear : function(name){
14972         delete this.state[name];
14973         this.fireEvent("statechange", this, name, null);
14974     },
14975     
14976     /**
14977      * Sets the value for a key
14978      * @param {String} name The key name
14979      * @param {Mixed} value The value to set
14980      */
14981     set : function(name, value){
14982         this.state[name] = value;
14983         this.fireEvent("statechange", this, name, value);
14984     },
14985     
14986     /**
14987      * Decodes a string previously encoded with {@link #encodeValue}.
14988      * @param {String} value The value to decode
14989      * @return {Mixed} The decoded value
14990      */
14991     decodeValue : function(cookie){
14992         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14993         var matches = re.exec(unescape(cookie));
14994         if(!matches || !matches[1]) {
14995             return; // non state cookie
14996         }
14997         var type = matches[1];
14998         var v = matches[2];
14999         switch(type){
15000             case "n":
15001                 return parseFloat(v);
15002             case "d":
15003                 return new Date(Date.parse(v));
15004             case "b":
15005                 return (v == "1");
15006             case "a":
15007                 var all = [];
15008                 var values = v.split("^");
15009                 for(var i = 0, len = values.length; i < len; i++){
15010                     all.push(this.decodeValue(values[i]));
15011                 }
15012                 return all;
15013            case "o":
15014                 var all = {};
15015                 var values = v.split("^");
15016                 for(var i = 0, len = values.length; i < len; i++){
15017                     var kv = values[i].split("=");
15018                     all[kv[0]] = this.decodeValue(kv[1]);
15019                 }
15020                 return all;
15021            default:
15022                 return v;
15023         }
15024     },
15025     
15026     /**
15027      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15028      * @param {Mixed} value The value to encode
15029      * @return {String} The encoded value
15030      */
15031     encodeValue : function(v){
15032         var enc;
15033         if(typeof v == "number"){
15034             enc = "n:" + v;
15035         }else if(typeof v == "boolean"){
15036             enc = "b:" + (v ? "1" : "0");
15037         }else if(v instanceof Date){
15038             enc = "d:" + v.toGMTString();
15039         }else if(v instanceof Array){
15040             var flat = "";
15041             for(var i = 0, len = v.length; i < len; i++){
15042                 flat += this.encodeValue(v[i]);
15043                 if(i != len-1) {
15044                     flat += "^";
15045                 }
15046             }
15047             enc = "a:" + flat;
15048         }else if(typeof v == "object"){
15049             var flat = "";
15050             for(var key in v){
15051                 if(typeof v[key] != "function"){
15052                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15053                 }
15054             }
15055             enc = "o:" + flat.substring(0, flat.length-1);
15056         }else{
15057             enc = "s:" + v;
15058         }
15059         return escape(enc);        
15060     }
15061 });
15062
15063 /*
15064  * Based on:
15065  * Ext JS Library 1.1.1
15066  * Copyright(c) 2006-2007, Ext JS, LLC.
15067  *
15068  * Originally Released Under LGPL - original licence link has changed is not relivant.
15069  *
15070  * Fork - LGPL
15071  * <script type="text/javascript">
15072  */
15073 /**
15074  * @class Roo.state.Manager
15075  * This is the global state manager. By default all components that are "state aware" check this class
15076  * for state information if you don't pass them a custom state provider. In order for this class
15077  * to be useful, it must be initialized with a provider when your application initializes.
15078  <pre><code>
15079 // in your initialization function
15080 init : function(){
15081    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15082    ...
15083    // supposed you have a {@link Roo.BorderLayout}
15084    var layout = new Roo.BorderLayout(...);
15085    layout.restoreState();
15086    // or a {Roo.BasicDialog}
15087    var dialog = new Roo.BasicDialog(...);
15088    dialog.restoreState();
15089  </code></pre>
15090  * @singleton
15091  */
15092 Roo.state.Manager = function(){
15093     var provider = new Roo.state.Provider();
15094     
15095     return {
15096         /**
15097          * Configures the default state provider for your application
15098          * @param {Provider} stateProvider The state provider to set
15099          */
15100         setProvider : function(stateProvider){
15101             provider = stateProvider;
15102         },
15103         
15104         /**
15105          * Returns the current value for a key
15106          * @param {String} name The key name
15107          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15108          * @return {Mixed} The state data
15109          */
15110         get : function(key, defaultValue){
15111             return provider.get(key, defaultValue);
15112         },
15113         
15114         /**
15115          * Sets the value for a key
15116          * @param {String} name The key name
15117          * @param {Mixed} value The state data
15118          */
15119          set : function(key, value){
15120             provider.set(key, value);
15121         },
15122         
15123         /**
15124          * Clears a value from the state
15125          * @param {String} name The key name
15126          */
15127         clear : function(key){
15128             provider.clear(key);
15129         },
15130         
15131         /**
15132          * Gets the currently configured state provider
15133          * @return {Provider} The state provider
15134          */
15135         getProvider : function(){
15136             return provider;
15137         }
15138     };
15139 }();
15140 /*
15141  * Based on:
15142  * Ext JS Library 1.1.1
15143  * Copyright(c) 2006-2007, Ext JS, LLC.
15144  *
15145  * Originally Released Under LGPL - original licence link has changed is not relivant.
15146  *
15147  * Fork - LGPL
15148  * <script type="text/javascript">
15149  */
15150 /**
15151  * @class Roo.state.CookieProvider
15152  * @extends Roo.state.Provider
15153  * The default Provider implementation which saves state via cookies.
15154  * <br />Usage:
15155  <pre><code>
15156    var cp = new Roo.state.CookieProvider({
15157        path: "/cgi-bin/",
15158        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15159        domain: "roojs.com"
15160    })
15161    Roo.state.Manager.setProvider(cp);
15162  </code></pre>
15163  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15164  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15165  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15166  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15167  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15168  * domain the page is running on including the 'www' like 'www.roojs.com')
15169  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15170  * @constructor
15171  * Create a new CookieProvider
15172  * @param {Object} config The configuration object
15173  */
15174 Roo.state.CookieProvider = function(config){
15175     Roo.state.CookieProvider.superclass.constructor.call(this);
15176     this.path = "/";
15177     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15178     this.domain = null;
15179     this.secure = false;
15180     Roo.apply(this, config);
15181     this.state = this.readCookies();
15182 };
15183
15184 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15185     // private
15186     set : function(name, value){
15187         if(typeof value == "undefined" || value === null){
15188             this.clear(name);
15189             return;
15190         }
15191         this.setCookie(name, value);
15192         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15193     },
15194
15195     // private
15196     clear : function(name){
15197         this.clearCookie(name);
15198         Roo.state.CookieProvider.superclass.clear.call(this, name);
15199     },
15200
15201     // private
15202     readCookies : function(){
15203         var cookies = {};
15204         var c = document.cookie + ";";
15205         var re = /\s?(.*?)=(.*?);/g;
15206         var matches;
15207         while((matches = re.exec(c)) != null){
15208             var name = matches[1];
15209             var value = matches[2];
15210             if(name && name.substring(0,3) == "ys-"){
15211                 cookies[name.substr(3)] = this.decodeValue(value);
15212             }
15213         }
15214         return cookies;
15215     },
15216
15217     // private
15218     setCookie : function(name, value){
15219         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15220            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15221            ((this.path == null) ? "" : ("; path=" + this.path)) +
15222            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15223            ((this.secure == true) ? "; secure" : "");
15224     },
15225
15226     // private
15227     clearCookie : function(name){
15228         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15229            ((this.path == null) ? "" : ("; path=" + this.path)) +
15230            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15231            ((this.secure == true) ? "; secure" : "");
15232     }
15233 });/*
15234  * Based on:
15235  * Ext JS Library 1.1.1
15236  * Copyright(c) 2006-2007, Ext JS, LLC.
15237  *
15238  * Originally Released Under LGPL - original licence link has changed is not relivant.
15239  *
15240  * Fork - LGPL
15241  * <script type="text/javascript">
15242  */
15243  
15244
15245 /**
15246  * @class Roo.ComponentMgr
15247  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15248  * @singleton
15249  */
15250 Roo.ComponentMgr = function(){
15251     var all = new Roo.util.MixedCollection();
15252
15253     return {
15254         /**
15255          * Registers a component.
15256          * @param {Roo.Component} c The component
15257          */
15258         register : function(c){
15259             all.add(c);
15260         },
15261
15262         /**
15263          * Unregisters a component.
15264          * @param {Roo.Component} c The component
15265          */
15266         unregister : function(c){
15267             all.remove(c);
15268         },
15269
15270         /**
15271          * Returns a component by id
15272          * @param {String} id The component id
15273          */
15274         get : function(id){
15275             return all.get(id);
15276         },
15277
15278         /**
15279          * Registers a function that will be called when a specified component is added to ComponentMgr
15280          * @param {String} id The component id
15281          * @param {Funtction} fn The callback function
15282          * @param {Object} scope The scope of the callback
15283          */
15284         onAvailable : function(id, fn, scope){
15285             all.on("add", function(index, o){
15286                 if(o.id == id){
15287                     fn.call(scope || o, o);
15288                     all.un("add", fn, scope);
15289                 }
15290             });
15291         }
15292     };
15293 }();/*
15294  * Based on:
15295  * Ext JS Library 1.1.1
15296  * Copyright(c) 2006-2007, Ext JS, LLC.
15297  *
15298  * Originally Released Under LGPL - original licence link has changed is not relivant.
15299  *
15300  * Fork - LGPL
15301  * <script type="text/javascript">
15302  */
15303  
15304 /**
15305  * @class Roo.Component
15306  * @extends Roo.util.Observable
15307  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15308  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15309  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15310  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15311  * All visual components (widgets) that require rendering into a layout should subclass Component.
15312  * @constructor
15313  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15314  * element and its id used as the component id.  If a string is passed, it is assumed to be the id of an existing element
15315  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15316  */
15317 Roo.Component = function(config){
15318     config = config || {};
15319     if(config.tagName || config.dom || typeof config == "string"){ // element object
15320         config = {el: config, id: config.id || config};
15321     }
15322     this.initialConfig = config;
15323
15324     Roo.apply(this, config);
15325     this.addEvents({
15326         /**
15327          * @event disable
15328          * Fires after the component is disabled.
15329              * @param {Roo.Component} this
15330              */
15331         disable : true,
15332         /**
15333          * @event enable
15334          * Fires after the component is enabled.
15335              * @param {Roo.Component} this
15336              */
15337         enable : true,
15338         /**
15339          * @event beforeshow
15340          * Fires before the component is shown.  Return false to stop the show.
15341              * @param {Roo.Component} this
15342              */
15343         beforeshow : true,
15344         /**
15345          * @event show
15346          * Fires after the component is shown.
15347              * @param {Roo.Component} this
15348              */
15349         show : true,
15350         /**
15351          * @event beforehide
15352          * Fires before the component is hidden. Return false to stop the hide.
15353              * @param {Roo.Component} this
15354              */
15355         beforehide : true,
15356         /**
15357          * @event hide
15358          * Fires after the component is hidden.
15359              * @param {Roo.Component} this
15360              */
15361         hide : true,
15362         /**
15363          * @event beforerender
15364          * Fires before the component is rendered. Return false to stop the render.
15365              * @param {Roo.Component} this
15366              */
15367         beforerender : true,
15368         /**
15369          * @event render
15370          * Fires after the component is rendered.
15371              * @param {Roo.Component} this
15372              */
15373         render : true,
15374         /**
15375          * @event beforedestroy
15376          * Fires before the component is destroyed. Return false to stop the destroy.
15377              * @param {Roo.Component} this
15378              */
15379         beforedestroy : true,
15380         /**
15381          * @event destroy
15382          * Fires after the component is destroyed.
15383              * @param {Roo.Component} this
15384              */
15385         destroy : true
15386     });
15387     if(!this.id){
15388         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15389     }
15390     Roo.ComponentMgr.register(this);
15391     Roo.Component.superclass.constructor.call(this);
15392     this.initComponent();
15393     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15394         this.render(this.renderTo);
15395         delete this.renderTo;
15396     }
15397 };
15398
15399 /** @private */
15400 Roo.Component.AUTO_ID = 1000;
15401
15402 Roo.extend(Roo.Component, Roo.util.Observable, {
15403     /**
15404      * @scope Roo.Component.prototype
15405      * @type {Boolean}
15406      * true if this component is hidden. Read-only.
15407      */
15408     hidden : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component is disabled. Read-only.
15412      */
15413     disabled : false,
15414     /**
15415      * @type {Boolean}
15416      * true if this component has been rendered. Read-only.
15417      */
15418     rendered : false,
15419     
15420     /** @cfg {String} disableClass
15421      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15422      */
15423     disabledClass : "x-item-disabled",
15424         /** @cfg {Boolean} allowDomMove
15425          * Whether the component can move the Dom node when rendering (defaults to true).
15426          */
15427     allowDomMove : true,
15428     /** @cfg {String} hideMode (display|visibility)
15429      * How this component should hidden. Supported values are
15430      * "visibility" (css visibility), "offsets" (negative offset position) and
15431      * "display" (css display) - defaults to "display".
15432      */
15433     hideMode: 'display',
15434
15435     /** @private */
15436     ctype : "Roo.Component",
15437
15438     /**
15439      * @cfg {String} actionMode 
15440      * which property holds the element that used for  hide() / show() / disable() / enable()
15441      * default is 'el' 
15442      */
15443     actionMode : "el",
15444
15445     /** @private */
15446     getActionEl : function(){
15447         return this[this.actionMode];
15448     },
15449
15450     initComponent : Roo.emptyFn,
15451     /**
15452      * If this is a lazy rendering component, render it to its container element.
15453      * @param {String/HTMLElement/Element} container (optional) The element this component should be rendered into. If it is being applied to existing markup, this should be left off.
15454      */
15455     render : function(container, position){
15456         
15457         if(this.rendered){
15458             return this;
15459         }
15460         
15461         if(this.fireEvent("beforerender", this) === false){
15462             return false;
15463         }
15464         
15465         if(!container && this.el){
15466             this.el = Roo.get(this.el);
15467             container = this.el.dom.parentNode;
15468             this.allowDomMove = false;
15469         }
15470         this.container = Roo.get(container);
15471         this.rendered = true;
15472         if(position !== undefined){
15473             if(typeof position == 'number'){
15474                 position = this.container.dom.childNodes[position];
15475             }else{
15476                 position = Roo.getDom(position);
15477             }
15478         }
15479         this.onRender(this.container, position || null);
15480         if(this.cls){
15481             this.el.addClass(this.cls);
15482             delete this.cls;
15483         }
15484         if(this.style){
15485             this.el.applyStyles(this.style);
15486             delete this.style;
15487         }
15488         this.fireEvent("render", this);
15489         this.afterRender(this.container);
15490         if(this.hidden){
15491             this.hide();
15492         }
15493         if(this.disabled){
15494             this.disable();
15495         }
15496
15497         return this;
15498         
15499     },
15500
15501     /** @private */
15502     // default function is not really useful
15503     onRender : function(ct, position){
15504         if(this.el){
15505             this.el = Roo.get(this.el);
15506             if(this.allowDomMove !== false){
15507                 ct.dom.insertBefore(this.el.dom, position);
15508             }
15509         }
15510     },
15511
15512     /** @private */
15513     getAutoCreate : function(){
15514         var cfg = typeof this.autoCreate == "object" ?
15515                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15516         if(this.id && !cfg.id){
15517             cfg.id = this.id;
15518         }
15519         return cfg;
15520     },
15521
15522     /** @private */
15523     afterRender : Roo.emptyFn,
15524
15525     /**
15526      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15527      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15528      */
15529     destroy : function(){
15530         if(this.fireEvent("beforedestroy", this) !== false){
15531             this.purgeListeners();
15532             this.beforeDestroy();
15533             if(this.rendered){
15534                 this.el.removeAllListeners();
15535                 this.el.remove();
15536                 if(this.actionMode == "container"){
15537                     this.container.remove();
15538                 }
15539             }
15540             this.onDestroy();
15541             Roo.ComponentMgr.unregister(this);
15542             this.fireEvent("destroy", this);
15543         }
15544     },
15545
15546         /** @private */
15547     beforeDestroy : function(){
15548
15549     },
15550
15551         /** @private */
15552         onDestroy : function(){
15553
15554     },
15555
15556     /**
15557      * Returns the underlying {@link Roo.Element}.
15558      * @return {Roo.Element} The element
15559      */
15560     getEl : function(){
15561         return this.el;
15562     },
15563
15564     /**
15565      * Returns the id of this component.
15566      * @return {String}
15567      */
15568     getId : function(){
15569         return this.id;
15570     },
15571
15572     /**
15573      * Try to focus this component.
15574      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15575      * @return {Roo.Component} this
15576      */
15577     focus : function(selectText){
15578         if(this.rendered){
15579             this.el.focus();
15580             if(selectText === true){
15581                 this.el.dom.select();
15582             }
15583         }
15584         return this;
15585     },
15586
15587     /** @private */
15588     blur : function(){
15589         if(this.rendered){
15590             this.el.blur();
15591         }
15592         return this;
15593     },
15594
15595     /**
15596      * Disable this component.
15597      * @return {Roo.Component} this
15598      */
15599     disable : function(){
15600         if(this.rendered){
15601             this.onDisable();
15602         }
15603         this.disabled = true;
15604         this.fireEvent("disable", this);
15605         return this;
15606     },
15607
15608         // private
15609     onDisable : function(){
15610         this.getActionEl().addClass(this.disabledClass);
15611         this.el.dom.disabled = true;
15612     },
15613
15614     /**
15615      * Enable this component.
15616      * @return {Roo.Component} this
15617      */
15618     enable : function(){
15619         if(this.rendered){
15620             this.onEnable();
15621         }
15622         this.disabled = false;
15623         this.fireEvent("enable", this);
15624         return this;
15625     },
15626
15627         // private
15628     onEnable : function(){
15629         this.getActionEl().removeClass(this.disabledClass);
15630         this.el.dom.disabled = false;
15631     },
15632
15633     /**
15634      * Convenience function for setting disabled/enabled by boolean.
15635      * @param {Boolean} disabled
15636      */
15637     setDisabled : function(disabled){
15638         this[disabled ? "disable" : "enable"]();
15639     },
15640
15641     /**
15642      * Show this component.
15643      * @return {Roo.Component} this
15644      */
15645     show: function(){
15646         if(this.fireEvent("beforeshow", this) !== false){
15647             this.hidden = false;
15648             if(this.rendered){
15649                 this.onShow();
15650             }
15651             this.fireEvent("show", this);
15652         }
15653         return this;
15654     },
15655
15656     // private
15657     onShow : function(){
15658         var ae = this.getActionEl();
15659         if(this.hideMode == 'visibility'){
15660             ae.dom.style.visibility = "visible";
15661         }else if(this.hideMode == 'offsets'){
15662             ae.removeClass('x-hidden');
15663         }else{
15664             ae.dom.style.display = "";
15665         }
15666     },
15667
15668     /**
15669      * Hide this component.
15670      * @return {Roo.Component} this
15671      */
15672     hide: function(){
15673         if(this.fireEvent("beforehide", this) !== false){
15674             this.hidden = true;
15675             if(this.rendered){
15676                 this.onHide();
15677             }
15678             this.fireEvent("hide", this);
15679         }
15680         return this;
15681     },
15682
15683     // private
15684     onHide : function(){
15685         var ae = this.getActionEl();
15686         if(this.hideMode == 'visibility'){
15687             ae.dom.style.visibility = "hidden";
15688         }else if(this.hideMode == 'offsets'){
15689             ae.addClass('x-hidden');
15690         }else{
15691             ae.dom.style.display = "none";
15692         }
15693     },
15694
15695     /**
15696      * Convenience function to hide or show this component by boolean.
15697      * @param {Boolean} visible True to show, false to hide
15698      * @return {Roo.Component} this
15699      */
15700     setVisible: function(visible){
15701         if(visible) {
15702             this.show();
15703         }else{
15704             this.hide();
15705         }
15706         return this;
15707     },
15708
15709     /**
15710      * Returns true if this component is visible.
15711      */
15712     isVisible : function(){
15713         return this.getActionEl().isVisible();
15714     },
15715
15716     cloneConfig : function(overrides){
15717         overrides = overrides || {};
15718         var id = overrides.id || Roo.id();
15719         var cfg = Roo.applyIf(overrides, this.initialConfig);
15720         cfg.id = id; // prevent dup id
15721         return new this.constructor(cfg);
15722     }
15723 });/*
15724  * Based on:
15725  * Ext JS Library 1.1.1
15726  * Copyright(c) 2006-2007, Ext JS, LLC.
15727  *
15728  * Originally Released Under LGPL - original licence link has changed is not relivant.
15729  *
15730  * Fork - LGPL
15731  * <script type="text/javascript">
15732  */
15733
15734 /**
15735  * @class Roo.BoxComponent
15736  * @extends Roo.Component
15737  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15738  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15739  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15740  * layout containers.
15741  * @constructor
15742  * @param {Roo.Element/String/Object} config The configuration options.
15743  */
15744 Roo.BoxComponent = function(config){
15745     Roo.Component.call(this, config);
15746     this.addEvents({
15747         /**
15748          * @event resize
15749          * Fires after the component is resized.
15750              * @param {Roo.Component} this
15751              * @param {Number} adjWidth The box-adjusted width that was set
15752              * @param {Number} adjHeight The box-adjusted height that was set
15753              * @param {Number} rawWidth The width that was originally specified
15754              * @param {Number} rawHeight The height that was originally specified
15755              */
15756         resize : true,
15757         /**
15758          * @event move
15759          * Fires after the component is moved.
15760              * @param {Roo.Component} this
15761              * @param {Number} x The new x position
15762              * @param {Number} y The new y position
15763              */
15764         move : true
15765     });
15766 };
15767
15768 Roo.extend(Roo.BoxComponent, Roo.Component, {
15769     // private, set in afterRender to signify that the component has been rendered
15770     boxReady : false,
15771     // private, used to defer height settings to subclasses
15772     deferHeight: false,
15773     /** @cfg {Number} width
15774      * width (optional) size of component
15775      */
15776      /** @cfg {Number} height
15777      * height (optional) size of component
15778      */
15779      
15780     /**
15781      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15782      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15783      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15784      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15785      * @return {Roo.BoxComponent} this
15786      */
15787     setSize : function(w, h){
15788         // support for standard size objects
15789         if(typeof w == 'object'){
15790             h = w.height;
15791             w = w.width;
15792         }
15793         // not rendered
15794         if(!this.boxReady){
15795             this.width = w;
15796             this.height = h;
15797             return this;
15798         }
15799
15800         // prevent recalcs when not needed
15801         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15802             return this;
15803         }
15804         this.lastSize = {width: w, height: h};
15805
15806         var adj = this.adjustSize(w, h);
15807         var aw = adj.width, ah = adj.height;
15808         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15809             var rz = this.getResizeEl();
15810             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15811                 rz.setSize(aw, ah);
15812             }else if(!this.deferHeight && ah !== undefined){
15813                 rz.setHeight(ah);
15814             }else if(aw !== undefined){
15815                 rz.setWidth(aw);
15816             }
15817             this.onResize(aw, ah, w, h);
15818             this.fireEvent('resize', this, aw, ah, w, h);
15819         }
15820         return this;
15821     },
15822
15823     /**
15824      * Gets the current size of the component's underlying element.
15825      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15826      */
15827     getSize : function(){
15828         return this.el.getSize();
15829     },
15830
15831     /**
15832      * Gets the current XY position of the component's underlying element.
15833      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15834      * @return {Array} The XY position of the element (e.g., [100, 200])
15835      */
15836     getPosition : function(local){
15837         if(local === true){
15838             return [this.el.getLeft(true), this.el.getTop(true)];
15839         }
15840         return this.xy || this.el.getXY();
15841     },
15842
15843     /**
15844      * Gets the current box measurements of the component's underlying element.
15845      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15846      * @returns {Object} box An object in the format {x, y, width, height}
15847      */
15848     getBox : function(local){
15849         var s = this.el.getSize();
15850         if(local){
15851             s.x = this.el.getLeft(true);
15852             s.y = this.el.getTop(true);
15853         }else{
15854             var xy = this.xy || this.el.getXY();
15855             s.x = xy[0];
15856             s.y = xy[1];
15857         }
15858         return s;
15859     },
15860
15861     /**
15862      * Sets the current box measurements of the component's underlying element.
15863      * @param {Object} box An object in the format {x, y, width, height}
15864      * @returns {Roo.BoxComponent} this
15865      */
15866     updateBox : function(box){
15867         this.setSize(box.width, box.height);
15868         this.setPagePosition(box.x, box.y);
15869         return this;
15870     },
15871
15872     // protected
15873     getResizeEl : function(){
15874         return this.resizeEl || this.el;
15875     },
15876
15877     // protected
15878     getPositionEl : function(){
15879         return this.positionEl || this.el;
15880     },
15881
15882     /**
15883      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15884      * This method fires the move event.
15885      * @param {Number} left The new left
15886      * @param {Number} top The new top
15887      * @returns {Roo.BoxComponent} this
15888      */
15889     setPosition : function(x, y){
15890         this.x = x;
15891         this.y = y;
15892         if(!this.boxReady){
15893             return this;
15894         }
15895         var adj = this.adjustPosition(x, y);
15896         var ax = adj.x, ay = adj.y;
15897
15898         var el = this.getPositionEl();
15899         if(ax !== undefined || ay !== undefined){
15900             if(ax !== undefined && ay !== undefined){
15901                 el.setLeftTop(ax, ay);
15902             }else if(ax !== undefined){
15903                 el.setLeft(ax);
15904             }else if(ay !== undefined){
15905                 el.setTop(ay);
15906             }
15907             this.onPosition(ax, ay);
15908             this.fireEvent('move', this, ax, ay);
15909         }
15910         return this;
15911     },
15912
15913     /**
15914      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15915      * This method fires the move event.
15916      * @param {Number} x The new x position
15917      * @param {Number} y The new y position
15918      * @returns {Roo.BoxComponent} this
15919      */
15920     setPagePosition : function(x, y){
15921         this.pageX = x;
15922         this.pageY = y;
15923         if(!this.boxReady){
15924             return;
15925         }
15926         if(x === undefined || y === undefined){ // cannot translate undefined points
15927             return;
15928         }
15929         var p = this.el.translatePoints(x, y);
15930         this.setPosition(p.left, p.top);
15931         return this;
15932     },
15933
15934     // private
15935     onRender : function(ct, position){
15936         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15937         if(this.resizeEl){
15938             this.resizeEl = Roo.get(this.resizeEl);
15939         }
15940         if(this.positionEl){
15941             this.positionEl = Roo.get(this.positionEl);
15942         }
15943     },
15944
15945     // private
15946     afterRender : function(){
15947         Roo.BoxComponent.superclass.afterRender.call(this);
15948         this.boxReady = true;
15949         this.setSize(this.width, this.height);
15950         if(this.x || this.y){
15951             this.setPosition(this.x, this.y);
15952         }
15953         if(this.pageX || this.pageY){
15954             this.setPagePosition(this.pageX, this.pageY);
15955         }
15956     },
15957
15958     /**
15959      * Force the component's size to recalculate based on the underlying element's current height and width.
15960      * @returns {Roo.BoxComponent} this
15961      */
15962     syncSize : function(){
15963         delete this.lastSize;
15964         this.setSize(this.el.getWidth(), this.el.getHeight());
15965         return this;
15966     },
15967
15968     /**
15969      * Called after the component is resized, this method is empty by default but can be implemented by any
15970      * subclass that needs to perform custom logic after a resize occurs.
15971      * @param {Number} adjWidth The box-adjusted width that was set
15972      * @param {Number} adjHeight The box-adjusted height that was set
15973      * @param {Number} rawWidth The width that was originally specified
15974      * @param {Number} rawHeight The height that was originally specified
15975      */
15976     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15977
15978     },
15979
15980     /**
15981      * Called after the component is moved, this method is empty by default but can be implemented by any
15982      * subclass that needs to perform custom logic after a move occurs.
15983      * @param {Number} x The new x position
15984      * @param {Number} y The new y position
15985      */
15986     onPosition : function(x, y){
15987
15988     },
15989
15990     // private
15991     adjustSize : function(w, h){
15992         if(this.autoWidth){
15993             w = 'auto';
15994         }
15995         if(this.autoHeight){
15996             h = 'auto';
15997         }
15998         return {width : w, height: h};
15999     },
16000
16001     // private
16002     adjustPosition : function(x, y){
16003         return {x : x, y: y};
16004     }
16005 });/*
16006  * Original code for Roojs - LGPL
16007  * <script type="text/javascript">
16008  */
16009  
16010 /**
16011  * @class Roo.XComponent
16012  * A delayed Element creator...
16013  * Or a way to group chunks of interface together.
16014  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16015  *  used in conjunction with XComponent.build() it will create an instance of each element,
16016  *  then call addxtype() to build the User interface.
16017  * 
16018  * Mypart.xyx = new Roo.XComponent({
16019
16020     parent : 'Mypart.xyz', // empty == document.element.!!
16021     order : '001',
16022     name : 'xxxx'
16023     region : 'xxxx'
16024     disabled : function() {} 
16025      
16026     tree : function() { // return an tree of xtype declared components
16027         var MODULE = this;
16028         return 
16029         {
16030             xtype : 'NestedLayoutPanel',
16031             // technicall
16032         }
16033      ]
16034  *})
16035  *
16036  *
16037  * It can be used to build a big heiracy, with parent etc.
16038  * or you can just use this to render a single compoent to a dom element
16039  * MYPART.render(Roo.Element | String(id) | dom_element )
16040  *
16041  *
16042  * Usage patterns.
16043  *
16044  * Classic Roo
16045  *
16046  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16047  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16048  *
16049  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16050  *
16051  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16052  * - if mulitple topModules exist, the last one is defined as the top module.
16053  *
16054  * Embeded Roo
16055  * 
16056  * When the top level or multiple modules are to embedded into a existing HTML page,
16057  * the parent element can container '#id' of the element where the module will be drawn.
16058  *
16059  * Bootstrap Roo
16060  *
16061  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16062  * it relies more on a include mechanism, where sub modules are included into an outer page.
16063  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16064  * 
16065  * Bootstrap Roo Included elements
16066  *
16067  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16068  * hence confusing the component builder as it thinks there are multiple top level elements. 
16069  *
16070  * String Over-ride & Translations
16071  *
16072  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16073  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16074  * are needed. @see Roo.XComponent.overlayString  
16075  * 
16076  * 
16077  * 
16078  * @extends Roo.util.Observable
16079  * @constructor
16080  * @param cfg {Object} configuration of component
16081  * 
16082  */
16083 Roo.XComponent = function(cfg) {
16084     Roo.apply(this, cfg);
16085     this.addEvents({ 
16086         /**
16087              * @event built
16088              * Fires when this the componnt is built
16089              * @param {Roo.XComponent} c the component
16090              */
16091         'built' : true
16092         
16093     });
16094     this.region = this.region || 'center'; // default..
16095     Roo.XComponent.register(this);
16096     this.modules = false;
16097     this.el = false; // where the layout goes..
16098     
16099     
16100 }
16101 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16102     /**
16103      * @property el
16104      * The created element (with Roo.factory())
16105      * @type {Roo.Layout}
16106      */
16107     el  : false,
16108     
16109     /**
16110      * @property el
16111      * for BC  - use el in new code
16112      * @type {Roo.Layout}
16113      */
16114     panel : false,
16115     
16116     /**
16117      * @property layout
16118      * for BC  - use el in new code
16119      * @type {Roo.Layout}
16120      */
16121     layout : false,
16122     
16123      /**
16124      * @cfg {Function|boolean} disabled
16125      * If this module is disabled by some rule, return true from the funtion
16126      */
16127     disabled : false,
16128     
16129     /**
16130      * @cfg {String} parent 
16131      * Name of parent element which it get xtype added to..
16132      */
16133     parent: false,
16134     
16135     /**
16136      * @cfg {String} order
16137      * Used to set the order in which elements are created (usefull for multiple tabs)
16138      */
16139     
16140     order : false,
16141     /**
16142      * @cfg {String} name
16143      * String to display while loading.
16144      */
16145     name : false,
16146     /**
16147      * @cfg {String} region
16148      * Region to render component to (defaults to center)
16149      */
16150     region : 'center',
16151     
16152     /**
16153      * @cfg {Array} items
16154      * A single item array - the first element is the root of the tree..
16155      * It's done this way to stay compatible with the Xtype system...
16156      */
16157     items : false,
16158     
16159     /**
16160      * @property _tree
16161      * The method that retuns the tree of parts that make up this compoennt 
16162      * @type {function}
16163      */
16164     _tree  : false,
16165     
16166      /**
16167      * render
16168      * render element to dom or tree
16169      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16170      */
16171     
16172     render : function(el)
16173     {
16174         
16175         el = el || false;
16176         var hp = this.parent ? 1 : 0;
16177         Roo.debug &&  Roo.log(this);
16178         
16179         var tree = this._tree ? this._tree() : this.tree();
16180
16181         
16182         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16183             // if parent is a '#.....' string, then let's use that..
16184             var ename = this.parent.substr(1);
16185             this.parent = false;
16186             Roo.debug && Roo.log(ename);
16187             switch (ename) {
16188                 case 'bootstrap-body':
16189                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16190                         // this is the BorderLayout standard?
16191                        this.parent = { el : true };
16192                        break;
16193                     }
16194                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16195                         // need to insert stuff...
16196                         this.parent =  {
16197                              el : new Roo.bootstrap.layout.Border({
16198                                  el : document.body, 
16199                      
16200                                  center: {
16201                                     titlebar: false,
16202                                     autoScroll:false,
16203                                     closeOnTab: true,
16204                                     tabPosition: 'top',
16205                                       //resizeTabs: true,
16206                                     alwaysShowTabs: true,
16207                                     hideTabs: false
16208                                      //minTabWidth: 140
16209                                  }
16210                              })
16211                         
16212                          };
16213                          break;
16214                     }
16215                          
16216                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16217                         this.parent = { el :  new  Roo.bootstrap.Body() };
16218                         Roo.debug && Roo.log("setting el to doc body");
16219                          
16220                     } else {
16221                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16222                     }
16223                     break;
16224                 case 'bootstrap':
16225                     this.parent = { el : true};
16226                     // fall through
16227                 default:
16228                     el = Roo.get(ename);
16229                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16230                         this.parent = { el : true};
16231                     }
16232                     
16233                     break;
16234             }
16235                 
16236             
16237             if (!el && !this.parent) {
16238                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16239                 return;
16240             }
16241         }
16242         
16243         Roo.debug && Roo.log("EL:");
16244         Roo.debug && Roo.log(el);
16245         Roo.debug && Roo.log("this.parent.el:");
16246         Roo.debug && Roo.log(this.parent.el);
16247         
16248
16249         // altertive root elements ??? - we need a better way to indicate these.
16250         var is_alt = Roo.XComponent.is_alt ||
16251                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16252                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16253                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16254         
16255         
16256         
16257         if (!this.parent && is_alt) {
16258             //el = Roo.get(document.body);
16259             this.parent = { el : true };
16260         }
16261             
16262             
16263         
16264         if (!this.parent) {
16265             
16266             Roo.debug && Roo.log("no parent - creating one");
16267             
16268             el = el ? Roo.get(el) : false;      
16269             
16270             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16271                 
16272                 this.parent =  {
16273                     el : new Roo.bootstrap.layout.Border({
16274                         el: el || document.body,
16275                     
16276                         center: {
16277                             titlebar: false,
16278                             autoScroll:false,
16279                             closeOnTab: true,
16280                             tabPosition: 'top',
16281                              //resizeTabs: true,
16282                             alwaysShowTabs: false,
16283                             hideTabs: true,
16284                             minTabWidth: 140,
16285                             overflow: 'visible'
16286                          }
16287                      })
16288                 };
16289             } else {
16290             
16291                 // it's a top level one..
16292                 this.parent =  {
16293                     el : new Roo.BorderLayout(el || document.body, {
16294                         center: {
16295                             titlebar: false,
16296                             autoScroll:false,
16297                             closeOnTab: true,
16298                             tabPosition: 'top',
16299                              //resizeTabs: true,
16300                             alwaysShowTabs: el && hp? false :  true,
16301                             hideTabs: el || !hp ? true :  false,
16302                             minTabWidth: 140
16303                          }
16304                     })
16305                 };
16306             }
16307         }
16308         
16309         if (!this.parent.el) {
16310                 // probably an old style ctor, which has been disabled.
16311                 return;
16312
16313         }
16314                 // The 'tree' method is  '_tree now' 
16315             
16316         tree.region = tree.region || this.region;
16317         var is_body = false;
16318         if (this.parent.el === true) {
16319             // bootstrap... - body..
16320             if (el) {
16321                 tree.el = el;
16322             }
16323             this.parent.el = Roo.factory(tree);
16324             is_body = true;
16325         }
16326         
16327         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16328         this.fireEvent('built', this);
16329         
16330         this.panel = this.el;
16331         this.layout = this.panel.layout;
16332         this.parentLayout = this.parent.layout  || false;  
16333          
16334     }
16335     
16336 });
16337
16338 Roo.apply(Roo.XComponent, {
16339     /**
16340      * @property  hideProgress
16341      * true to disable the building progress bar.. usefull on single page renders.
16342      * @type Boolean
16343      */
16344     hideProgress : false,
16345     /**
16346      * @property  buildCompleted
16347      * True when the builder has completed building the interface.
16348      * @type Boolean
16349      */
16350     buildCompleted : false,
16351      
16352     /**
16353      * @property  topModule
16354      * the upper most module - uses document.element as it's constructor.
16355      * @type Object
16356      */
16357      
16358     topModule  : false,
16359       
16360     /**
16361      * @property  modules
16362      * array of modules to be created by registration system.
16363      * @type {Array} of Roo.XComponent
16364      */
16365     
16366     modules : [],
16367     /**
16368      * @property  elmodules
16369      * array of modules to be created by which use #ID 
16370      * @type {Array} of Roo.XComponent
16371      */
16372      
16373     elmodules : [],
16374
16375      /**
16376      * @property  is_alt
16377      * Is an alternative Root - normally used by bootstrap or other systems,
16378      *    where the top element in the tree can wrap 'body' 
16379      * @type {boolean}  (default false)
16380      */
16381      
16382     is_alt : false,
16383     /**
16384      * @property  build_from_html
16385      * Build elements from html - used by bootstrap HTML stuff 
16386      *    - this is cleared after build is completed
16387      * @type {boolean}    (default false)
16388      */
16389      
16390     build_from_html : false,
16391     /**
16392      * Register components to be built later.
16393      *
16394      * This solves the following issues
16395      * - Building is not done on page load, but after an authentication process has occured.
16396      * - Interface elements are registered on page load
16397      * - Parent Interface elements may not be loaded before child, so this handles that..
16398      * 
16399      *
16400      * example:
16401      * 
16402      * MyApp.register({
16403           order : '000001',
16404           module : 'Pman.Tab.projectMgr',
16405           region : 'center',
16406           parent : 'Pman.layout',
16407           disabled : false,  // or use a function..
16408         })
16409      
16410      * * @param {Object} details about module
16411      */
16412     register : function(obj) {
16413                 
16414         Roo.XComponent.event.fireEvent('register', obj);
16415         switch(typeof(obj.disabled) ) {
16416                 
16417             case 'undefined':
16418                 break;
16419             
16420             case 'function':
16421                 if ( obj.disabled() ) {
16422                         return;
16423                 }
16424                 break;
16425             
16426             default:
16427                 if (obj.disabled || obj.region == '#disabled') {
16428                         return;
16429                 }
16430                 break;
16431         }
16432                 
16433         this.modules.push(obj);
16434          
16435     },
16436     /**
16437      * convert a string to an object..
16438      * eg. 'AAA.BBB' -> finds AAA.BBB
16439
16440      */
16441     
16442     toObject : function(str)
16443     {
16444         if (!str || typeof(str) == 'object') {
16445             return str;
16446         }
16447         if (str.substring(0,1) == '#') {
16448             return str;
16449         }
16450
16451         var ar = str.split('.');
16452         var rt, o;
16453         rt = ar.shift();
16454             /** eval:var:o */
16455         try {
16456             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16457         } catch (e) {
16458             throw "Module not found : " + str;
16459         }
16460         
16461         if (o === false) {
16462             throw "Module not found : " + str;
16463         }
16464         Roo.each(ar, function(e) {
16465             if (typeof(o[e]) == 'undefined') {
16466                 throw "Module not found : " + str;
16467             }
16468             o = o[e];
16469         });
16470         
16471         return o;
16472         
16473     },
16474     
16475     
16476     /**
16477      * move modules into their correct place in the tree..
16478      * 
16479      */
16480     preBuild : function ()
16481     {
16482         var _t = this;
16483         Roo.each(this.modules , function (obj)
16484         {
16485             Roo.XComponent.event.fireEvent('beforebuild', obj);
16486             
16487             var opar = obj.parent;
16488             try { 
16489                 obj.parent = this.toObject(opar);
16490             } catch(e) {
16491                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16492                 return;
16493             }
16494             
16495             if (!obj.parent) {
16496                 Roo.debug && Roo.log("GOT top level module");
16497                 Roo.debug && Roo.log(obj);
16498                 obj.modules = new Roo.util.MixedCollection(false, 
16499                     function(o) { return o.order + '' }
16500                 );
16501                 this.topModule = obj;
16502                 return;
16503             }
16504                         // parent is a string (usually a dom element name..)
16505             if (typeof(obj.parent) == 'string') {
16506                 this.elmodules.push(obj);
16507                 return;
16508             }
16509             if (obj.parent.constructor != Roo.XComponent) {
16510                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16511             }
16512             if (!obj.parent.modules) {
16513                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16514                     function(o) { return o.order + '' }
16515                 );
16516             }
16517             if (obj.parent.disabled) {
16518                 obj.disabled = true;
16519             }
16520             obj.parent.modules.add(obj);
16521         }, this);
16522     },
16523     
16524      /**
16525      * make a list of modules to build.
16526      * @return {Array} list of modules. 
16527      */ 
16528     
16529     buildOrder : function()
16530     {
16531         var _this = this;
16532         var cmp = function(a,b) {   
16533             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16534         };
16535         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16536             throw "No top level modules to build";
16537         }
16538         
16539         // make a flat list in order of modules to build.
16540         var mods = this.topModule ? [ this.topModule ] : [];
16541                 
16542         
16543         // elmodules (is a list of DOM based modules )
16544         Roo.each(this.elmodules, function(e) {
16545             mods.push(e);
16546             if (!this.topModule &&
16547                 typeof(e.parent) == 'string' &&
16548                 e.parent.substring(0,1) == '#' &&
16549                 Roo.get(e.parent.substr(1))
16550                ) {
16551                 
16552                 _this.topModule = e;
16553             }
16554             
16555         });
16556
16557         
16558         // add modules to their parents..
16559         var addMod = function(m) {
16560             Roo.debug && Roo.log("build Order: add: " + m.name);
16561                 
16562             mods.push(m);
16563             if (m.modules && !m.disabled) {
16564                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16565                 m.modules.keySort('ASC',  cmp );
16566                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16567     
16568                 m.modules.each(addMod);
16569             } else {
16570                 Roo.debug && Roo.log("build Order: no child modules");
16571             }
16572             // not sure if this is used any more..
16573             if (m.finalize) {
16574                 m.finalize.name = m.name + " (clean up) ";
16575                 mods.push(m.finalize);
16576             }
16577             
16578         }
16579         if (this.topModule && this.topModule.modules) { 
16580             this.topModule.modules.keySort('ASC',  cmp );
16581             this.topModule.modules.each(addMod);
16582         } 
16583         return mods;
16584     },
16585     
16586      /**
16587      * Build the registered modules.
16588      * @param {Object} parent element.
16589      * @param {Function} optional method to call after module has been added.
16590      * 
16591      */ 
16592    
16593     build : function(opts) 
16594     {
16595         
16596         if (typeof(opts) != 'undefined') {
16597             Roo.apply(this,opts);
16598         }
16599         
16600         this.preBuild();
16601         var mods = this.buildOrder();
16602       
16603         //this.allmods = mods;
16604         //Roo.debug && Roo.log(mods);
16605         //return;
16606         if (!mods.length) { // should not happen
16607             throw "NO modules!!!";
16608         }
16609         
16610         
16611         var msg = "Building Interface...";
16612         // flash it up as modal - so we store the mask!?
16613         if (!this.hideProgress && Roo.MessageBox) {
16614             Roo.MessageBox.show({ title: 'loading' });
16615             Roo.MessageBox.show({
16616                title: "Please wait...",
16617                msg: msg,
16618                width:450,
16619                progress:true,
16620                buttons : false,
16621                closable:false,
16622                modal: false
16623               
16624             });
16625         }
16626         var total = mods.length;
16627         
16628         var _this = this;
16629         var progressRun = function() {
16630             if (!mods.length) {
16631                 Roo.debug && Roo.log('hide?');
16632                 if (!this.hideProgress && Roo.MessageBox) {
16633                     Roo.MessageBox.hide();
16634                 }
16635                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16636                 
16637                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16638                 
16639                 // THE END...
16640                 return false;   
16641             }
16642             
16643             var m = mods.shift();
16644             
16645             
16646             Roo.debug && Roo.log(m);
16647             // not sure if this is supported any more.. - modules that are are just function
16648             if (typeof(m) == 'function') { 
16649                 m.call(this);
16650                 return progressRun.defer(10, _this);
16651             } 
16652             
16653             
16654             msg = "Building Interface " + (total  - mods.length) + 
16655                     " of " + total + 
16656                     (m.name ? (' - ' + m.name) : '');
16657                         Roo.debug && Roo.log(msg);
16658             if (!_this.hideProgress &&  Roo.MessageBox) { 
16659                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16660             }
16661             
16662          
16663             // is the module disabled?
16664             var disabled = (typeof(m.disabled) == 'function') ?
16665                 m.disabled.call(m.module.disabled) : m.disabled;    
16666             
16667             
16668             if (disabled) {
16669                 return progressRun(); // we do not update the display!
16670             }
16671             
16672             // now build 
16673             
16674                         
16675                         
16676             m.render();
16677             // it's 10 on top level, and 1 on others??? why...
16678             return progressRun.defer(10, _this);
16679              
16680         }
16681         progressRun.defer(1, _this);
16682      
16683         
16684         
16685     },
16686     /**
16687      * Overlay a set of modified strings onto a component
16688      * This is dependant on our builder exporting the strings and 'named strings' elements.
16689      * 
16690      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16691      * @param {Object} associative array of 'named' string and it's new value.
16692      * 
16693      */
16694         overlayStrings : function( component, strings )
16695     {
16696         if (typeof(component['_named_strings']) == 'undefined') {
16697             throw "ERROR: component does not have _named_strings";
16698         }
16699         for ( var k in strings ) {
16700             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16701             if (md !== false) {
16702                 component['_strings'][md] = strings[k];
16703             } else {
16704                 Roo.log('could not find named string: ' + k + ' in');
16705                 Roo.log(component);
16706             }
16707             
16708         }
16709         
16710     },
16711     
16712         
16713         /**
16714          * Event Object.
16715          *
16716          *
16717          */
16718         event: false, 
16719     /**
16720          * wrapper for event.on - aliased later..  
16721          * Typically use to register a event handler for register:
16722          *
16723          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16724          *
16725          */
16726     on : false
16727    
16728     
16729     
16730 });
16731
16732 Roo.XComponent.event = new Roo.util.Observable({
16733                 events : { 
16734                         /**
16735                          * @event register
16736                          * Fires when an Component is registered,
16737                          * set the disable property on the Component to stop registration.
16738                          * @param {Roo.XComponent} c the component being registerd.
16739                          * 
16740                          */
16741                         'register' : true,
16742             /**
16743                          * @event beforebuild
16744                          * Fires before each Component is built
16745                          * can be used to apply permissions.
16746                          * @param {Roo.XComponent} c the component being registerd.
16747                          * 
16748                          */
16749                         'beforebuild' : true,
16750                         /**
16751                          * @event buildcomplete
16752                          * Fires on the top level element when all elements have been built
16753                          * @param {Roo.XComponent} the top level component.
16754                          */
16755                         'buildcomplete' : true
16756                         
16757                 }
16758 });
16759
16760 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16761  //
16762  /**
16763  * marked - a markdown parser
16764  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16765  * https://github.com/chjj/marked
16766  */
16767
16768
16769 /**
16770  *
16771  * Roo.Markdown - is a very crude wrapper around marked..
16772  *
16773  * usage:
16774  * 
16775  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16776  * 
16777  * Note: move the sample code to the bottom of this
16778  * file before uncommenting it.
16779  *
16780  */
16781
16782 Roo.Markdown = {};
16783 Roo.Markdown.toHtml = function(text) {
16784     
16785     var c = new Roo.Markdown.marked.setOptions({
16786             renderer: new Roo.Markdown.marked.Renderer(),
16787             gfm: true,
16788             tables: true,
16789             breaks: false,
16790             pedantic: false,
16791             sanitize: false,
16792             smartLists: true,
16793             smartypants: false
16794           });
16795     // A FEW HACKS!!?
16796     
16797     text = text.replace(/\\\n/g,' ');
16798     return Roo.Markdown.marked(text);
16799 };
16800 //
16801 // converter
16802 //
16803 // Wraps all "globals" so that the only thing
16804 // exposed is makeHtml().
16805 //
16806 (function() {
16807     
16808      /**
16809          * eval:var:escape
16810          * eval:var:unescape
16811          * eval:var:replace
16812          */
16813       
16814     /**
16815      * Helpers
16816      */
16817     
16818     var escape = function (html, encode) {
16819       return html
16820         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
16821         .replace(/</g, '&lt;')
16822         .replace(/>/g, '&gt;')
16823         .replace(/"/g, '&quot;')
16824         .replace(/'/g, '&#39;');
16825     }
16826     
16827     var unescape = function (html) {
16828         // explicitly match decimal, hex, and named HTML entities 
16829       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
16830         n = n.toLowerCase();
16831         if (n === 'colon') { return ':'; }
16832         if (n.charAt(0) === '#') {
16833           return n.charAt(1) === 'x'
16834             ? String.fromCharCode(parseInt(n.substring(2), 16))
16835             : String.fromCharCode(+n.substring(1));
16836         }
16837         return '';
16838       });
16839     }
16840     
16841     var replace = function (regex, opt) {
16842       regex = regex.source;
16843       opt = opt || '';
16844       return function self(name, val) {
16845         if (!name) { return new RegExp(regex, opt); }
16846         val = val.source || val;
16847         val = val.replace(/(^|[^\[])\^/g, '$1');
16848         regex = regex.replace(name, val);
16849         return self;
16850       };
16851     }
16852
16853
16854          /**
16855          * eval:var:noop
16856     */
16857     var noop = function () {}
16858     noop.exec = noop;
16859     
16860          /**
16861          * eval:var:merge
16862     */
16863     var merge = function (obj) {
16864       var i = 1
16865         , target
16866         , key;
16867     
16868       for (; i < arguments.length; i++) {
16869         target = arguments[i];
16870         for (key in target) {
16871           if (Object.prototype.hasOwnProperty.call(target, key)) {
16872             obj[key] = target[key];
16873           }
16874         }
16875       }
16876     
16877       return obj;
16878     }
16879     
16880     
16881     /**
16882      * Block-Level Grammar
16883      */
16884     
16885     
16886     
16887     
16888     var block = {
16889       newline: /^\n+/,
16890       code: /^( {4}[^\n]+\n*)+/,
16891       fences: noop,
16892       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16893       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16894       nptable: noop,
16895       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16896       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16897       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16898       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16899       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16900       table: noop,
16901       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16902       text: /^[^\n]+/
16903     };
16904     
16905     block.bullet = /(?:[*+-]|\d+\.)/;
16906     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16907     block.item = replace(block.item, 'gm')
16908       (/bull/g, block.bullet)
16909       ();
16910     
16911     block.list = replace(block.list)
16912       (/bull/g, block.bullet)
16913       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16914       ('def', '\\n+(?=' + block.def.source + ')')
16915       ();
16916     
16917     block.blockquote = replace(block.blockquote)
16918       ('def', block.def)
16919       ();
16920     
16921     block._tag = '(?!(?:'
16922       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16923       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16924       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16925     
16926     block.html = replace(block.html)
16927       ('comment', /<!--[\s\S]*?-->/)
16928       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16929       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16930       (/tag/g, block._tag)
16931       ();
16932     
16933     block.paragraph = replace(block.paragraph)
16934       ('hr', block.hr)
16935       ('heading', block.heading)
16936       ('lheading', block.lheading)
16937       ('blockquote', block.blockquote)
16938       ('tag', '<' + block._tag)
16939       ('def', block.def)
16940       ();
16941     
16942     /**
16943      * Normal Block Grammar
16944      */
16945     
16946     block.normal = merge({}, block);
16947     
16948     /**
16949      * GFM Block Grammar
16950      */
16951     
16952     block.gfm = merge({}, block.normal, {
16953       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16954       paragraph: /^/,
16955       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16956     });
16957     
16958     block.gfm.paragraph = replace(block.paragraph)
16959       ('(?!', '(?!'
16960         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16961         + block.list.source.replace('\\1', '\\3') + '|')
16962       ();
16963     
16964     /**
16965      * GFM + Tables Block Grammar
16966      */
16967     
16968     block.tables = merge({}, block.gfm, {
16969       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16970       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16971     });
16972     
16973     /**
16974      * Block Lexer
16975      */
16976     
16977     var Lexer = function (options) {
16978       this.tokens = [];
16979       this.tokens.links = {};
16980       this.options = options || marked.defaults;
16981       this.rules = block.normal;
16982     
16983       if (this.options.gfm) {
16984         if (this.options.tables) {
16985           this.rules = block.tables;
16986         } else {
16987           this.rules = block.gfm;
16988         }
16989       }
16990     }
16991     
16992     /**
16993      * Expose Block Rules
16994      */
16995     
16996     Lexer.rules = block;
16997     
16998     /**
16999      * Static Lex Method
17000      */
17001     
17002     Lexer.lex = function(src, options) {
17003       var lexer = new Lexer(options);
17004       return lexer.lex(src);
17005     };
17006     
17007     /**
17008      * Preprocessing
17009      */
17010     
17011     Lexer.prototype.lex = function(src) {
17012       src = src
17013         .replace(/\r\n|\r/g, '\n')
17014         .replace(/\t/g, '    ')
17015         .replace(/\u00a0/g, ' ')
17016         .replace(/\u2424/g, '\n');
17017     
17018       return this.token(src, true);
17019     };
17020     
17021     /**
17022      * Lexing
17023      */
17024     
17025     Lexer.prototype.token = function(src, top, bq) {
17026       var src = src.replace(/^ +$/gm, '')
17027         , next
17028         , loose
17029         , cap
17030         , bull
17031         , b
17032         , item
17033         , space
17034         , i
17035         , l;
17036     
17037       while (src) {
17038         // newline
17039         if (cap = this.rules.newline.exec(src)) {
17040           src = src.substring(cap[0].length);
17041           if (cap[0].length > 1) {
17042             this.tokens.push({
17043               type: 'space'
17044             });
17045           }
17046         }
17047     
17048         // code
17049         if (cap = this.rules.code.exec(src)) {
17050           src = src.substring(cap[0].length);
17051           cap = cap[0].replace(/^ {4}/gm, '');
17052           this.tokens.push({
17053             type: 'code',
17054             text: !this.options.pedantic
17055               ? cap.replace(/\n+$/, '')
17056               : cap
17057           });
17058           continue;
17059         }
17060     
17061         // fences (gfm)
17062         if (cap = this.rules.fences.exec(src)) {
17063           src = src.substring(cap[0].length);
17064           this.tokens.push({
17065             type: 'code',
17066             lang: cap[2],
17067             text: cap[3] || ''
17068           });
17069           continue;
17070         }
17071     
17072         // heading
17073         if (cap = this.rules.heading.exec(src)) {
17074           src = src.substring(cap[0].length);
17075           this.tokens.push({
17076             type: 'heading',
17077             depth: cap[1].length,
17078             text: cap[2]
17079           });
17080           continue;
17081         }
17082     
17083         // table no leading pipe (gfm)
17084         if (top && (cap = this.rules.nptable.exec(src))) {
17085           src = src.substring(cap[0].length);
17086     
17087           item = {
17088             type: 'table',
17089             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17090             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17091             cells: cap[3].replace(/\n$/, '').split('\n')
17092           };
17093     
17094           for (i = 0; i < item.align.length; i++) {
17095             if (/^ *-+: *$/.test(item.align[i])) {
17096               item.align[i] = 'right';
17097             } else if (/^ *:-+: *$/.test(item.align[i])) {
17098               item.align[i] = 'center';
17099             } else if (/^ *:-+ *$/.test(item.align[i])) {
17100               item.align[i] = 'left';
17101             } else {
17102               item.align[i] = null;
17103             }
17104           }
17105     
17106           for (i = 0; i < item.cells.length; i++) {
17107             item.cells[i] = item.cells[i].split(/ *\| */);
17108           }
17109     
17110           this.tokens.push(item);
17111     
17112           continue;
17113         }
17114     
17115         // lheading
17116         if (cap = this.rules.lheading.exec(src)) {
17117           src = src.substring(cap[0].length);
17118           this.tokens.push({
17119             type: 'heading',
17120             depth: cap[2] === '=' ? 1 : 2,
17121             text: cap[1]
17122           });
17123           continue;
17124         }
17125     
17126         // hr
17127         if (cap = this.rules.hr.exec(src)) {
17128           src = src.substring(cap[0].length);
17129           this.tokens.push({
17130             type: 'hr'
17131           });
17132           continue;
17133         }
17134     
17135         // blockquote
17136         if (cap = this.rules.blockquote.exec(src)) {
17137           src = src.substring(cap[0].length);
17138     
17139           this.tokens.push({
17140             type: 'blockquote_start'
17141           });
17142     
17143           cap = cap[0].replace(/^ *> ?/gm, '');
17144     
17145           // Pass `top` to keep the current
17146           // "toplevel" state. This is exactly
17147           // how markdown.pl works.
17148           this.token(cap, top, true);
17149     
17150           this.tokens.push({
17151             type: 'blockquote_end'
17152           });
17153     
17154           continue;
17155         }
17156     
17157         // list
17158         if (cap = this.rules.list.exec(src)) {
17159           src = src.substring(cap[0].length);
17160           bull = cap[2];
17161     
17162           this.tokens.push({
17163             type: 'list_start',
17164             ordered: bull.length > 1
17165           });
17166     
17167           // Get each top-level item.
17168           cap = cap[0].match(this.rules.item);
17169     
17170           next = false;
17171           l = cap.length;
17172           i = 0;
17173     
17174           for (; i < l; i++) {
17175             item = cap[i];
17176     
17177             // Remove the list item's bullet
17178             // so it is seen as the next token.
17179             space = item.length;
17180             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17181     
17182             // Outdent whatever the
17183             // list item contains. Hacky.
17184             if (~item.indexOf('\n ')) {
17185               space -= item.length;
17186               item = !this.options.pedantic
17187                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17188                 : item.replace(/^ {1,4}/gm, '');
17189             }
17190     
17191             // Determine whether the next list item belongs here.
17192             // Backpedal if it does not belong in this list.
17193             if (this.options.smartLists && i !== l - 1) {
17194               b = block.bullet.exec(cap[i + 1])[0];
17195               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17196                 src = cap.slice(i + 1).join('\n') + src;
17197                 i = l - 1;
17198               }
17199             }
17200     
17201             // Determine whether item is loose or not.
17202             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17203             // for discount behavior.
17204             loose = next || /\n\n(?!\s*$)/.test(item);
17205             if (i !== l - 1) {
17206               next = item.charAt(item.length - 1) === '\n';
17207               if (!loose) { loose = next; }
17208             }
17209     
17210             this.tokens.push({
17211               type: loose
17212                 ? 'loose_item_start'
17213                 : 'list_item_start'
17214             });
17215     
17216             // Recurse.
17217             this.token(item, false, bq);
17218     
17219             this.tokens.push({
17220               type: 'list_item_end'
17221             });
17222           }
17223     
17224           this.tokens.push({
17225             type: 'list_end'
17226           });
17227     
17228           continue;
17229         }
17230     
17231         // html
17232         if (cap = this.rules.html.exec(src)) {
17233           src = src.substring(cap[0].length);
17234           this.tokens.push({
17235             type: this.options.sanitize
17236               ? 'paragraph'
17237               : 'html',
17238             pre: !this.options.sanitizer
17239               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17240             text: cap[0]
17241           });
17242           continue;
17243         }
17244     
17245         // def
17246         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17247           src = src.substring(cap[0].length);
17248           this.tokens.links[cap[1].toLowerCase()] = {
17249             href: cap[2],
17250             title: cap[3]
17251           };
17252           continue;
17253         }
17254     
17255         // table (gfm)
17256         if (top && (cap = this.rules.table.exec(src))) {
17257           src = src.substring(cap[0].length);
17258     
17259           item = {
17260             type: 'table',
17261             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17262             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17263             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17264           };
17265     
17266           for (i = 0; i < item.align.length; i++) {
17267             if (/^ *-+: *$/.test(item.align[i])) {
17268               item.align[i] = 'right';
17269             } else if (/^ *:-+: *$/.test(item.align[i])) {
17270               item.align[i] = 'center';
17271             } else if (/^ *:-+ *$/.test(item.align[i])) {
17272               item.align[i] = 'left';
17273             } else {
17274               item.align[i] = null;
17275             }
17276           }
17277     
17278           for (i = 0; i < item.cells.length; i++) {
17279             item.cells[i] = item.cells[i]
17280               .replace(/^ *\| *| *\| *$/g, '')
17281               .split(/ *\| */);
17282           }
17283     
17284           this.tokens.push(item);
17285     
17286           continue;
17287         }
17288     
17289         // top-level paragraph
17290         if (top && (cap = this.rules.paragraph.exec(src))) {
17291           src = src.substring(cap[0].length);
17292           this.tokens.push({
17293             type: 'paragraph',
17294             text: cap[1].charAt(cap[1].length - 1) === '\n'
17295               ? cap[1].slice(0, -1)
17296               : cap[1]
17297           });
17298           continue;
17299         }
17300     
17301         // text
17302         if (cap = this.rules.text.exec(src)) {
17303           // Top-level should never reach here.
17304           src = src.substring(cap[0].length);
17305           this.tokens.push({
17306             type: 'text',
17307             text: cap[0]
17308           });
17309           continue;
17310         }
17311     
17312         if (src) {
17313           throw new
17314             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17315         }
17316       }
17317     
17318       return this.tokens;
17319     };
17320     
17321     /**
17322      * Inline-Level Grammar
17323      */
17324     
17325     var inline = {
17326       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17327       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17328       url: noop,
17329       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17330       link: /^!?\[(inside)\]\(href\)/,
17331       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17332       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17333       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17334       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17335       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17336       br: /^ {2,}\n(?!\s*$)/,
17337       del: noop,
17338       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17339     };
17340     
17341     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17342     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17343     
17344     inline.link = replace(inline.link)
17345       ('inside', inline._inside)
17346       ('href', inline._href)
17347       ();
17348     
17349     inline.reflink = replace(inline.reflink)
17350       ('inside', inline._inside)
17351       ();
17352     
17353     /**
17354      * Normal Inline Grammar
17355      */
17356     
17357     inline.normal = merge({}, inline);
17358     
17359     /**
17360      * Pedantic Inline Grammar
17361      */
17362     
17363     inline.pedantic = merge({}, inline.normal, {
17364       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17365       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17366     });
17367     
17368     /**
17369      * GFM Inline Grammar
17370      */
17371     
17372     inline.gfm = merge({}, inline.normal, {
17373       escape: replace(inline.escape)('])', '~|])')(),
17374       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17375       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17376       text: replace(inline.text)
17377         (']|', '~]|')
17378         ('|', '|https?://|')
17379         ()
17380     });
17381     
17382     /**
17383      * GFM + Line Breaks Inline Grammar
17384      */
17385     
17386     inline.breaks = merge({}, inline.gfm, {
17387       br: replace(inline.br)('{2,}', '*')(),
17388       text: replace(inline.gfm.text)('{2,}', '*')()
17389     });
17390     
17391     /**
17392      * Inline Lexer & Compiler
17393      */
17394     
17395     var InlineLexer  = function (links, options) {
17396       this.options = options || marked.defaults;
17397       this.links = links;
17398       this.rules = inline.normal;
17399       this.renderer = this.options.renderer || new Renderer;
17400       this.renderer.options = this.options;
17401     
17402       if (!this.links) {
17403         throw new
17404           Error('Tokens array requires a `links` property.');
17405       }
17406     
17407       if (this.options.gfm) {
17408         if (this.options.breaks) {
17409           this.rules = inline.breaks;
17410         } else {
17411           this.rules = inline.gfm;
17412         }
17413       } else if (this.options.pedantic) {
17414         this.rules = inline.pedantic;
17415       }
17416     }
17417     
17418     /**
17419      * Expose Inline Rules
17420      */
17421     
17422     InlineLexer.rules = inline;
17423     
17424     /**
17425      * Static Lexing/Compiling Method
17426      */
17427     
17428     InlineLexer.output = function(src, links, options) {
17429       var inline = new InlineLexer(links, options);
17430       return inline.output(src);
17431     };
17432     
17433     /**
17434      * Lexing/Compiling
17435      */
17436     
17437     InlineLexer.prototype.output = function(src) {
17438       var out = ''
17439         , link
17440         , text
17441         , href
17442         , cap;
17443     
17444       while (src) {
17445         // escape
17446         if (cap = this.rules.escape.exec(src)) {
17447           src = src.substring(cap[0].length);
17448           out += cap[1];
17449           continue;
17450         }
17451     
17452         // autolink
17453         if (cap = this.rules.autolink.exec(src)) {
17454           src = src.substring(cap[0].length);
17455           if (cap[2] === '@') {
17456             text = cap[1].charAt(6) === ':'
17457               ? this.mangle(cap[1].substring(7))
17458               : this.mangle(cap[1]);
17459             href = this.mangle('mailto:') + text;
17460           } else {
17461             text = escape(cap[1]);
17462             href = text;
17463           }
17464           out += this.renderer.link(href, null, text);
17465           continue;
17466         }
17467     
17468         // url (gfm)
17469         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17470           src = src.substring(cap[0].length);
17471           text = escape(cap[1]);
17472           href = text;
17473           out += this.renderer.link(href, null, text);
17474           continue;
17475         }
17476     
17477         // tag
17478         if (cap = this.rules.tag.exec(src)) {
17479           if (!this.inLink && /^<a /i.test(cap[0])) {
17480             this.inLink = true;
17481           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17482             this.inLink = false;
17483           }
17484           src = src.substring(cap[0].length);
17485           out += this.options.sanitize
17486             ? this.options.sanitizer
17487               ? this.options.sanitizer(cap[0])
17488               : escape(cap[0])
17489             : cap[0];
17490           continue;
17491         }
17492     
17493         // link
17494         if (cap = this.rules.link.exec(src)) {
17495           src = src.substring(cap[0].length);
17496           this.inLink = true;
17497           out += this.outputLink(cap, {
17498             href: cap[2],
17499             title: cap[3]
17500           });
17501           this.inLink = false;
17502           continue;
17503         }
17504     
17505         // reflink, nolink
17506         if ((cap = this.rules.reflink.exec(src))
17507             || (cap = this.rules.nolink.exec(src))) {
17508           src = src.substring(cap[0].length);
17509           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17510           link = this.links[link.toLowerCase()];
17511           if (!link || !link.href) {
17512             out += cap[0].charAt(0);
17513             src = cap[0].substring(1) + src;
17514             continue;
17515           }
17516           this.inLink = true;
17517           out += this.outputLink(cap, link);
17518           this.inLink = false;
17519           continue;
17520         }
17521     
17522         // strong
17523         if (cap = this.rules.strong.exec(src)) {
17524           src = src.substring(cap[0].length);
17525           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17526           continue;
17527         }
17528     
17529         // em
17530         if (cap = this.rules.em.exec(src)) {
17531           src = src.substring(cap[0].length);
17532           out += this.renderer.em(this.output(cap[2] || cap[1]));
17533           continue;
17534         }
17535     
17536         // code
17537         if (cap = this.rules.code.exec(src)) {
17538           src = src.substring(cap[0].length);
17539           out += this.renderer.codespan(escape(cap[2], true));
17540           continue;
17541         }
17542     
17543         // br
17544         if (cap = this.rules.br.exec(src)) {
17545           src = src.substring(cap[0].length);
17546           out += this.renderer.br();
17547           continue;
17548         }
17549     
17550         // del (gfm)
17551         if (cap = this.rules.del.exec(src)) {
17552           src = src.substring(cap[0].length);
17553           out += this.renderer.del(this.output(cap[1]));
17554           continue;
17555         }
17556     
17557         // text
17558         if (cap = this.rules.text.exec(src)) {
17559           src = src.substring(cap[0].length);
17560           out += this.renderer.text(escape(this.smartypants(cap[0])));
17561           continue;
17562         }
17563     
17564         if (src) {
17565           throw new
17566             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17567         }
17568       }
17569     
17570       return out;
17571     };
17572     
17573     /**
17574      * Compile Link
17575      */
17576     
17577     InlineLexer.prototype.outputLink = function(cap, link) {
17578       var href = escape(link.href)
17579         , title = link.title ? escape(link.title) : null;
17580     
17581       return cap[0].charAt(0) !== '!'
17582         ? this.renderer.link(href, title, this.output(cap[1]))
17583         : this.renderer.image(href, title, escape(cap[1]));
17584     };
17585     
17586     /**
17587      * Smartypants Transformations
17588      */
17589     
17590     InlineLexer.prototype.smartypants = function(text) {
17591       if (!this.options.smartypants)  { return text; }
17592       return text
17593         // em-dashes
17594         .replace(/---/g, '\u2014')
17595         // en-dashes
17596         .replace(/--/g, '\u2013')
17597         // opening singles
17598         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17599         // closing singles & apostrophes
17600         .replace(/'/g, '\u2019')
17601         // opening doubles
17602         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17603         // closing doubles
17604         .replace(/"/g, '\u201d')
17605         // ellipses
17606         .replace(/\.{3}/g, '\u2026');
17607     };
17608     
17609     /**
17610      * Mangle Links
17611      */
17612     
17613     InlineLexer.prototype.mangle = function(text) {
17614       if (!this.options.mangle) { return text; }
17615       var out = ''
17616         , l = text.length
17617         , i = 0
17618         , ch;
17619     
17620       for (; i < l; i++) {
17621         ch = text.charCodeAt(i);
17622         if (Math.random() > 0.5) {
17623           ch = 'x' + ch.toString(16);
17624         }
17625         out += '&#' + ch + ';';
17626       }
17627     
17628       return out;
17629     };
17630     
17631     /**
17632      * Renderer
17633      */
17634     
17635      /**
17636          * eval:var:Renderer
17637     */
17638     
17639     var Renderer   = function (options) {
17640       this.options = options || {};
17641     }
17642     
17643     Renderer.prototype.code = function(code, lang, escaped) {
17644       if (this.options.highlight) {
17645         var out = this.options.highlight(code, lang);
17646         if (out != null && out !== code) {
17647           escaped = true;
17648           code = out;
17649         }
17650       } else {
17651             // hack!!! - it's already escapeD?
17652             escaped = true;
17653       }
17654     
17655       if (!lang) {
17656         return '<pre><code>'
17657           + (escaped ? code : escape(code, true))
17658           + '\n</code></pre>';
17659       }
17660     
17661       return '<pre><code class="'
17662         + this.options.langPrefix
17663         + escape(lang, true)
17664         + '">'
17665         + (escaped ? code : escape(code, true))
17666         + '\n</code></pre>\n';
17667     };
17668     
17669     Renderer.prototype.blockquote = function(quote) {
17670       return '<blockquote>\n' + quote + '</blockquote>\n';
17671     };
17672     
17673     Renderer.prototype.html = function(html) {
17674       return html;
17675     };
17676     
17677     Renderer.prototype.heading = function(text, level, raw) {
17678       return '<h'
17679         + level
17680         + ' id="'
17681         + this.options.headerPrefix
17682         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17683         + '">'
17684         + text
17685         + '</h'
17686         + level
17687         + '>\n';
17688     };
17689     
17690     Renderer.prototype.hr = function() {
17691       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17692     };
17693     
17694     Renderer.prototype.list = function(body, ordered) {
17695       var type = ordered ? 'ol' : 'ul';
17696       return '<' + type + '>\n' + body + '</' + type + '>\n';
17697     };
17698     
17699     Renderer.prototype.listitem = function(text) {
17700       return '<li>' + text + '</li>\n';
17701     };
17702     
17703     Renderer.prototype.paragraph = function(text) {
17704       return '<p>' + text + '</p>\n';
17705     };
17706     
17707     Renderer.prototype.table = function(header, body) {
17708       return '<table class="table table-striped">\n'
17709         + '<thead>\n'
17710         + header
17711         + '</thead>\n'
17712         + '<tbody>\n'
17713         + body
17714         + '</tbody>\n'
17715         + '</table>\n';
17716     };
17717     
17718     Renderer.prototype.tablerow = function(content) {
17719       return '<tr>\n' + content + '</tr>\n';
17720     };
17721     
17722     Renderer.prototype.tablecell = function(content, flags) {
17723       var type = flags.header ? 'th' : 'td';
17724       var tag = flags.align
17725         ? '<' + type + ' style="text-align:' + flags.align + '">'
17726         : '<' + type + '>';
17727       return tag + content + '</' + type + '>\n';
17728     };
17729     
17730     // span level renderer
17731     Renderer.prototype.strong = function(text) {
17732       return '<strong>' + text + '</strong>';
17733     };
17734     
17735     Renderer.prototype.em = function(text) {
17736       return '<em>' + text + '</em>';
17737     };
17738     
17739     Renderer.prototype.codespan = function(text) {
17740       return '<code>' + text + '</code>';
17741     };
17742     
17743     Renderer.prototype.br = function() {
17744       return this.options.xhtml ? '<br/>' : '<br>';
17745     };
17746     
17747     Renderer.prototype.del = function(text) {
17748       return '<del>' + text + '</del>';
17749     };
17750     
17751     Renderer.prototype.link = function(href, title, text) {
17752       if (this.options.sanitize) {
17753         try {
17754           var prot = decodeURIComponent(unescape(href))
17755             .replace(/[^\w:]/g, '')
17756             .toLowerCase();
17757         } catch (e) {
17758           return '';
17759         }
17760         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17761           return '';
17762         }
17763       }
17764       var out = '<a href="' + href + '"';
17765       if (title) {
17766         out += ' title="' + title + '"';
17767       }
17768       out += '>' + text + '</a>';
17769       return out;
17770     };
17771     
17772     Renderer.prototype.image = function(href, title, text) {
17773       var out = '<img src="' + href + '" alt="' + text + '"';
17774       if (title) {
17775         out += ' title="' + title + '"';
17776       }
17777       out += this.options.xhtml ? '/>' : '>';
17778       return out;
17779     };
17780     
17781     Renderer.prototype.text = function(text) {
17782       return text;
17783     };
17784     
17785     /**
17786      * Parsing & Compiling
17787      */
17788          /**
17789          * eval:var:Parser
17790     */
17791     
17792     var Parser= function (options) {
17793       this.tokens = [];
17794       this.token = null;
17795       this.options = options || marked.defaults;
17796       this.options.renderer = this.options.renderer || new Renderer;
17797       this.renderer = this.options.renderer;
17798       this.renderer.options = this.options;
17799     }
17800     
17801     /**
17802      * Static Parse Method
17803      */
17804     
17805     Parser.parse = function(src, options, renderer) {
17806       var parser = new Parser(options, renderer);
17807       return parser.parse(src);
17808     };
17809     
17810     /**
17811      * Parse Loop
17812      */
17813     
17814     Parser.prototype.parse = function(src) {
17815       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17816       this.tokens = src.reverse();
17817     
17818       var out = '';
17819       while (this.next()) {
17820         out += this.tok();
17821       }
17822     
17823       return out;
17824     };
17825     
17826     /**
17827      * Next Token
17828      */
17829     
17830     Parser.prototype.next = function() {
17831       return this.token = this.tokens.pop();
17832     };
17833     
17834     /**
17835      * Preview Next Token
17836      */
17837     
17838     Parser.prototype.peek = function() {
17839       return this.tokens[this.tokens.length - 1] || 0;
17840     };
17841     
17842     /**
17843      * Parse Text Tokens
17844      */
17845     
17846     Parser.prototype.parseText = function() {
17847       var body = this.token.text;
17848     
17849       while (this.peek().type === 'text') {
17850         body += '\n' + this.next().text;
17851       }
17852     
17853       return this.inline.output(body);
17854     };
17855     
17856     /**
17857      * Parse Current Token
17858      */
17859     
17860     Parser.prototype.tok = function() {
17861       switch (this.token.type) {
17862         case 'space': {
17863           return '';
17864         }
17865         case 'hr': {
17866           return this.renderer.hr();
17867         }
17868         case 'heading': {
17869           return this.renderer.heading(
17870             this.inline.output(this.token.text),
17871             this.token.depth,
17872             this.token.text);
17873         }
17874         case 'code': {
17875           return this.renderer.code(this.token.text,
17876             this.token.lang,
17877             this.token.escaped);
17878         }
17879         case 'table': {
17880           var header = ''
17881             , body = ''
17882             , i
17883             , row
17884             , cell
17885             , flags
17886             , j;
17887     
17888           // header
17889           cell = '';
17890           for (i = 0; i < this.token.header.length; i++) {
17891             flags = { header: true, align: this.token.align[i] };
17892             cell += this.renderer.tablecell(
17893               this.inline.output(this.token.header[i]),
17894               { header: true, align: this.token.align[i] }
17895             );
17896           }
17897           header += this.renderer.tablerow(cell);
17898     
17899           for (i = 0; i < this.token.cells.length; i++) {
17900             row = this.token.cells[i];
17901     
17902             cell = '';
17903             for (j = 0; j < row.length; j++) {
17904               cell += this.renderer.tablecell(
17905                 this.inline.output(row[j]),
17906                 { header: false, align: this.token.align[j] }
17907               );
17908             }
17909     
17910             body += this.renderer.tablerow(cell);
17911           }
17912           return this.renderer.table(header, body);
17913         }
17914         case 'blockquote_start': {
17915           var body = '';
17916     
17917           while (this.next().type !== 'blockquote_end') {
17918             body += this.tok();
17919           }
17920     
17921           return this.renderer.blockquote(body);
17922         }
17923         case 'list_start': {
17924           var body = ''
17925             , ordered = this.token.ordered;
17926     
17927           while (this.next().type !== 'list_end') {
17928             body += this.tok();
17929           }
17930     
17931           return this.renderer.list(body, ordered);
17932         }
17933         case 'list_item_start': {
17934           var body = '';
17935     
17936           while (this.next().type !== 'list_item_end') {
17937             body += this.token.type === 'text'
17938               ? this.parseText()
17939               : this.tok();
17940           }
17941     
17942           return this.renderer.listitem(body);
17943         }
17944         case 'loose_item_start': {
17945           var body = '';
17946     
17947           while (this.next().type !== 'list_item_end') {
17948             body += this.tok();
17949           }
17950     
17951           return this.renderer.listitem(body);
17952         }
17953         case 'html': {
17954           var html = !this.token.pre && !this.options.pedantic
17955             ? this.inline.output(this.token.text)
17956             : this.token.text;
17957           return this.renderer.html(html);
17958         }
17959         case 'paragraph': {
17960           return this.renderer.paragraph(this.inline.output(this.token.text));
17961         }
17962         case 'text': {
17963           return this.renderer.paragraph(this.parseText());
17964         }
17965       }
17966     };
17967   
17968     
17969     /**
17970      * Marked
17971      */
17972          /**
17973          * eval:var:marked
17974     */
17975     var marked = function (src, opt, callback) {
17976       if (callback || typeof opt === 'function') {
17977         if (!callback) {
17978           callback = opt;
17979           opt = null;
17980         }
17981     
17982         opt = merge({}, marked.defaults, opt || {});
17983     
17984         var highlight = opt.highlight
17985           , tokens
17986           , pending
17987           , i = 0;
17988     
17989         try {
17990           tokens = Lexer.lex(src, opt)
17991         } catch (e) {
17992           return callback(e);
17993         }
17994     
17995         pending = tokens.length;
17996          /**
17997          * eval:var:done
17998     */
17999         var done = function(err) {
18000           if (err) {
18001             opt.highlight = highlight;
18002             return callback(err);
18003           }
18004     
18005           var out;
18006     
18007           try {
18008             out = Parser.parse(tokens, opt);
18009           } catch (e) {
18010             err = e;
18011           }
18012     
18013           opt.highlight = highlight;
18014     
18015           return err
18016             ? callback(err)
18017             : callback(null, out);
18018         };
18019     
18020         if (!highlight || highlight.length < 3) {
18021           return done();
18022         }
18023     
18024         delete opt.highlight;
18025     
18026         if (!pending) { return done(); }
18027     
18028         for (; i < tokens.length; i++) {
18029           (function(token) {
18030             if (token.type !== 'code') {
18031               return --pending || done();
18032             }
18033             return highlight(token.text, token.lang, function(err, code) {
18034               if (err) { return done(err); }
18035               if (code == null || code === token.text) {
18036                 return --pending || done();
18037               }
18038               token.text = code;
18039               token.escaped = true;
18040               --pending || done();
18041             });
18042           })(tokens[i]);
18043         }
18044     
18045         return;
18046       }
18047       try {
18048         if (opt) { opt = merge({}, marked.defaults, opt); }
18049         return Parser.parse(Lexer.lex(src, opt), opt);
18050       } catch (e) {
18051         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18052         if ((opt || marked.defaults).silent) {
18053           return '<p>An error occured:</p><pre>'
18054             + escape(e.message + '', true)
18055             + '</pre>';
18056         }
18057         throw e;
18058       }
18059     }
18060     
18061     /**
18062      * Options
18063      */
18064     
18065     marked.options =
18066     marked.setOptions = function(opt) {
18067       merge(marked.defaults, opt);
18068       return marked;
18069     };
18070     
18071     marked.defaults = {
18072       gfm: true,
18073       tables: true,
18074       breaks: false,
18075       pedantic: false,
18076       sanitize: false,
18077       sanitizer: null,
18078       mangle: true,
18079       smartLists: false,
18080       silent: false,
18081       highlight: null,
18082       langPrefix: 'lang-',
18083       smartypants: false,
18084       headerPrefix: '',
18085       renderer: new Renderer,
18086       xhtml: false
18087     };
18088     
18089     /**
18090      * Expose
18091      */
18092     
18093     marked.Parser = Parser;
18094     marked.parser = Parser.parse;
18095     
18096     marked.Renderer = Renderer;
18097     
18098     marked.Lexer = Lexer;
18099     marked.lexer = Lexer.lex;
18100     
18101     marked.InlineLexer = InlineLexer;
18102     marked.inlineLexer = InlineLexer.output;
18103     
18104     marked.parse = marked;
18105     
18106     Roo.Markdown.marked = marked;
18107
18108 })();/*
18109  * Based on:
18110  * Ext JS Library 1.1.1
18111  * Copyright(c) 2006-2007, Ext JS, LLC.
18112  *
18113  * Originally Released Under LGPL - original licence link has changed is not relivant.
18114  *
18115  * Fork - LGPL
18116  * <script type="text/javascript">
18117  */
18118
18119
18120
18121 /*
18122  * These classes are derivatives of the similarly named classes in the YUI Library.
18123  * The original license:
18124  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18125  * Code licensed under the BSD License:
18126  * http://developer.yahoo.net/yui/license.txt
18127  */
18128
18129 (function() {
18130
18131 var Event=Roo.EventManager;
18132 var Dom=Roo.lib.Dom;
18133
18134 /**
18135  * @class Roo.dd.DragDrop
18136  * @extends Roo.util.Observable
18137  * Defines the interface and base operation of items that that can be
18138  * dragged or can be drop targets.  It was designed to be extended, overriding
18139  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18140  * Up to three html elements can be associated with a DragDrop instance:
18141  * <ul>
18142  * <li>linked element: the element that is passed into the constructor.
18143  * This is the element which defines the boundaries for interaction with
18144  * other DragDrop objects.</li>
18145  * <li>handle element(s): The drag operation only occurs if the element that
18146  * was clicked matches a handle element.  By default this is the linked
18147  * element, but there are times that you will want only a portion of the
18148  * linked element to initiate the drag operation, and the setHandleElId()
18149  * method provides a way to define this.</li>
18150  * <li>drag element: this represents the element that would be moved along
18151  * with the cursor during a drag operation.  By default, this is the linked
18152  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18153  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18154  * </li>
18155  * </ul>
18156  * This class should not be instantiated until the onload event to ensure that
18157  * the associated elements are available.
18158  * The following would define a DragDrop obj that would interact with any
18159  * other DragDrop obj in the "group1" group:
18160  * <pre>
18161  *  dd = new Roo.dd.DragDrop("div1", "group1");
18162  * </pre>
18163  * Since none of the event handlers have been implemented, nothing would
18164  * actually happen if you were to run the code above.  Normally you would
18165  * override this class or one of the default implementations, but you can
18166  * also override the methods you want on an instance of the class...
18167  * <pre>
18168  *  dd.onDragDrop = function(e, id) {
18169  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18170  *  }
18171  * </pre>
18172  * @constructor
18173  * @param {String} id of the element that is linked to this instance
18174  * @param {String} sGroup the group of related DragDrop objects
18175  * @param {object} config an object containing configurable attributes
18176  *                Valid properties for DragDrop:
18177  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18178  */
18179 Roo.dd.DragDrop = function(id, sGroup, config) {
18180     if (id) {
18181         this.init(id, sGroup, config);
18182     }
18183     
18184 };
18185
18186 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18187
18188     /**
18189      * The id of the element associated with this object.  This is what we
18190      * refer to as the "linked element" because the size and position of
18191      * this element is used to determine when the drag and drop objects have
18192      * interacted.
18193      * @property id
18194      * @type String
18195      */
18196     id: null,
18197
18198     /**
18199      * Configuration attributes passed into the constructor
18200      * @property config
18201      * @type object
18202      */
18203     config: null,
18204
18205     /**
18206      * The id of the element that will be dragged.  By default this is same
18207      * as the linked element , but could be changed to another element. Ex:
18208      * Roo.dd.DDProxy
18209      * @property dragElId
18210      * @type String
18211      * @private
18212      */
18213     dragElId: null,
18214
18215     /**
18216      * the id of the element that initiates the drag operation.  By default
18217      * this is the linked element, but could be changed to be a child of this
18218      * element.  This lets us do things like only starting the drag when the
18219      * header element within the linked html element is clicked.
18220      * @property handleElId
18221      * @type String
18222      * @private
18223      */
18224     handleElId: null,
18225
18226     /**
18227      * An associative array of HTML tags that will be ignored if clicked.
18228      * @property invalidHandleTypes
18229      * @type {string: string}
18230      */
18231     invalidHandleTypes: null,
18232
18233     /**
18234      * An associative array of ids for elements that will be ignored if clicked
18235      * @property invalidHandleIds
18236      * @type {string: string}
18237      */
18238     invalidHandleIds: null,
18239
18240     /**
18241      * An indexted array of css class names for elements that will be ignored
18242      * if clicked.
18243      * @property invalidHandleClasses
18244      * @type string[]
18245      */
18246     invalidHandleClasses: null,
18247
18248     /**
18249      * The linked element's absolute X position at the time the drag was
18250      * started
18251      * @property startPageX
18252      * @type int
18253      * @private
18254      */
18255     startPageX: 0,
18256
18257     /**
18258      * The linked element's absolute X position at the time the drag was
18259      * started
18260      * @property startPageY
18261      * @type int
18262      * @private
18263      */
18264     startPageY: 0,
18265
18266     /**
18267      * The group defines a logical collection of DragDrop objects that are
18268      * related.  Instances only get events when interacting with other
18269      * DragDrop object in the same group.  This lets us define multiple
18270      * groups using a single DragDrop subclass if we want.
18271      * @property groups
18272      * @type {string: string}
18273      */
18274     groups: null,
18275
18276     /**
18277      * Individual drag/drop instances can be locked.  This will prevent
18278      * onmousedown start drag.
18279      * @property locked
18280      * @type boolean
18281      * @private
18282      */
18283     locked: false,
18284
18285     /**
18286      * Lock this instance
18287      * @method lock
18288      */
18289     lock: function() { this.locked = true; },
18290
18291     /**
18292      * Unlock this instace
18293      * @method unlock
18294      */
18295     unlock: function() { this.locked = false; },
18296
18297     /**
18298      * By default, all insances can be a drop target.  This can be disabled by
18299      * setting isTarget to false.
18300      * @method isTarget
18301      * @type boolean
18302      */
18303     isTarget: true,
18304
18305     /**
18306      * The padding configured for this drag and drop object for calculating
18307      * the drop zone intersection with this object.
18308      * @method padding
18309      * @type int[]
18310      */
18311     padding: null,
18312
18313     /**
18314      * Cached reference to the linked element
18315      * @property _domRef
18316      * @private
18317      */
18318     _domRef: null,
18319
18320     /**
18321      * Internal typeof flag
18322      * @property __ygDragDrop
18323      * @private
18324      */
18325     __ygDragDrop: true,
18326
18327     /**
18328      * Set to true when horizontal contraints are applied
18329      * @property constrainX
18330      * @type boolean
18331      * @private
18332      */
18333     constrainX: false,
18334
18335     /**
18336      * Set to true when vertical contraints are applied
18337      * @property constrainY
18338      * @type boolean
18339      * @private
18340      */
18341     constrainY: false,
18342
18343     /**
18344      * The left constraint
18345      * @property minX
18346      * @type int
18347      * @private
18348      */
18349     minX: 0,
18350
18351     /**
18352      * The right constraint
18353      * @property maxX
18354      * @type int
18355      * @private
18356      */
18357     maxX: 0,
18358
18359     /**
18360      * The up constraint
18361      * @property minY
18362      * @type int
18363      * @type int
18364      * @private
18365      */
18366     minY: 0,
18367
18368     /**
18369      * The down constraint
18370      * @property maxY
18371      * @type int
18372      * @private
18373      */
18374     maxY: 0,
18375
18376     /**
18377      * Maintain offsets when we resetconstraints.  Set to true when you want
18378      * the position of the element relative to its parent to stay the same
18379      * when the page changes
18380      *
18381      * @property maintainOffset
18382      * @type boolean
18383      */
18384     maintainOffset: false,
18385
18386     /**
18387      * Array of pixel locations the element will snap to if we specified a
18388      * horizontal graduation/interval.  This array is generated automatically
18389      * when you define a tick interval.
18390      * @property xTicks
18391      * @type int[]
18392      */
18393     xTicks: null,
18394
18395     /**
18396      * Array of pixel locations the element will snap to if we specified a
18397      * vertical graduation/interval.  This array is generated automatically
18398      * when you define a tick interval.
18399      * @property yTicks
18400      * @type int[]
18401      */
18402     yTicks: null,
18403
18404     /**
18405      * By default the drag and drop instance will only respond to the primary
18406      * button click (left button for a right-handed mouse).  Set to true to
18407      * allow drag and drop to start with any mouse click that is propogated
18408      * by the browser
18409      * @property primaryButtonOnly
18410      * @type boolean
18411      */
18412     primaryButtonOnly: true,
18413
18414     /**
18415      * The availabe property is false until the linked dom element is accessible.
18416      * @property available
18417      * @type boolean
18418      */
18419     available: false,
18420
18421     /**
18422      * By default, drags can only be initiated if the mousedown occurs in the
18423      * region the linked element is.  This is done in part to work around a
18424      * bug in some browsers that mis-report the mousedown if the previous
18425      * mouseup happened outside of the window.  This property is set to true
18426      * if outer handles are defined.
18427      *
18428      * @property hasOuterHandles
18429      * @type boolean
18430      * @default false
18431      */
18432     hasOuterHandles: false,
18433
18434     /**
18435      * Code that executes immediately before the startDrag event
18436      * @method b4StartDrag
18437      * @private
18438      */
18439     b4StartDrag: function(x, y) { },
18440
18441     /**
18442      * Abstract method called after a drag/drop object is clicked
18443      * and the drag or mousedown time thresholds have beeen met.
18444      * @method startDrag
18445      * @param {int} X click location
18446      * @param {int} Y click location
18447      */
18448     startDrag: function(x, y) { /* override this */ },
18449
18450     /**
18451      * Code that executes immediately before the onDrag event
18452      * @method b4Drag
18453      * @private
18454      */
18455     b4Drag: function(e) { },
18456
18457     /**
18458      * Abstract method called during the onMouseMove event while dragging an
18459      * object.
18460      * @method onDrag
18461      * @param {Event} e the mousemove event
18462      */
18463     onDrag: function(e) { /* override this */ },
18464
18465     /**
18466      * Abstract method called when this element fist begins hovering over
18467      * another DragDrop obj
18468      * @method onDragEnter
18469      * @param {Event} e the mousemove event
18470      * @param {String|DragDrop[]} id In POINT mode, the element
18471      * id this is hovering over.  In INTERSECT mode, an array of one or more
18472      * dragdrop items being hovered over.
18473      */
18474     onDragEnter: function(e, id) { /* override this */ },
18475
18476     /**
18477      * Code that executes immediately before the onDragOver event
18478      * @method b4DragOver
18479      * @private
18480      */
18481     b4DragOver: function(e) { },
18482
18483     /**
18484      * Abstract method called when this element is hovering over another
18485      * DragDrop obj
18486      * @method onDragOver
18487      * @param {Event} e the mousemove event
18488      * @param {String|DragDrop[]} id In POINT mode, the element
18489      * id this is hovering over.  In INTERSECT mode, an array of dd items
18490      * being hovered over.
18491      */
18492     onDragOver: function(e, id) { /* override this */ },
18493
18494     /**
18495      * Code that executes immediately before the onDragOut event
18496      * @method b4DragOut
18497      * @private
18498      */
18499     b4DragOut: function(e) { },
18500
18501     /**
18502      * Abstract method called when we are no longer hovering over an element
18503      * @method onDragOut
18504      * @param {Event} e the mousemove event
18505      * @param {String|DragDrop[]} id In POINT mode, the element
18506      * id this was hovering over.  In INTERSECT mode, an array of dd items
18507      * that the mouse is no longer over.
18508      */
18509     onDragOut: function(e, id) { /* override this */ },
18510
18511     /**
18512      * Code that executes immediately before the onDragDrop event
18513      * @method b4DragDrop
18514      * @private
18515      */
18516     b4DragDrop: function(e) { },
18517
18518     /**
18519      * Abstract method called when this item is dropped on another DragDrop
18520      * obj
18521      * @method onDragDrop
18522      * @param {Event} e the mouseup event
18523      * @param {String|DragDrop[]} id In POINT mode, the element
18524      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18525      * was dropped on.
18526      */
18527     onDragDrop: function(e, id) { /* override this */ },
18528
18529     /**
18530      * Abstract method called when this item is dropped on an area with no
18531      * drop target
18532      * @method onInvalidDrop
18533      * @param {Event} e the mouseup event
18534      */
18535     onInvalidDrop: function(e) { /* override this */ },
18536
18537     /**
18538      * Code that executes immediately before the endDrag event
18539      * @method b4EndDrag
18540      * @private
18541      */
18542     b4EndDrag: function(e) { },
18543
18544     /**
18545      * Fired when we are done dragging the object
18546      * @method endDrag
18547      * @param {Event} e the mouseup event
18548      */
18549     endDrag: function(e) { /* override this */ },
18550
18551     /**
18552      * Code executed immediately before the onMouseDown event
18553      * @method b4MouseDown
18554      * @param {Event} e the mousedown event
18555      * @private
18556      */
18557     b4MouseDown: function(e) {  },
18558
18559     /**
18560      * Event handler that fires when a drag/drop obj gets a mousedown
18561      * @method onMouseDown
18562      * @param {Event} e the mousedown event
18563      */
18564     onMouseDown: function(e) { /* override this */ },
18565
18566     /**
18567      * Event handler that fires when a drag/drop obj gets a mouseup
18568      * @method onMouseUp
18569      * @param {Event} e the mouseup event
18570      */
18571     onMouseUp: function(e) { /* override this */ },
18572
18573     /**
18574      * Override the onAvailable method to do what is needed after the initial
18575      * position was determined.
18576      * @method onAvailable
18577      */
18578     onAvailable: function () {
18579     },
18580
18581     /*
18582      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18583      * @type Object
18584      */
18585     defaultPadding : {left:0, right:0, top:0, bottom:0},
18586
18587     /*
18588      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18589  *
18590  * Usage:
18591  <pre><code>
18592  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18593                 { dragElId: "existingProxyDiv" });
18594  dd.startDrag = function(){
18595      this.constrainTo("parent-id");
18596  };
18597  </code></pre>
18598  * Or you can initalize it using the {@link Roo.Element} object:
18599  <pre><code>
18600  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18601      startDrag : function(){
18602          this.constrainTo("parent-id");
18603      }
18604  });
18605  </code></pre>
18606      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18607      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18608      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18609      * an object containing the sides to pad. For example: {right:10, bottom:10}
18610      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18611      */
18612     constrainTo : function(constrainTo, pad, inContent){
18613         if(typeof pad == "number"){
18614             pad = {left: pad, right:pad, top:pad, bottom:pad};
18615         }
18616         pad = pad || this.defaultPadding;
18617         var b = Roo.get(this.getEl()).getBox();
18618         var ce = Roo.get(constrainTo);
18619         var s = ce.getScroll();
18620         var c, cd = ce.dom;
18621         if(cd == document.body){
18622             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18623         }else{
18624             xy = ce.getXY();
18625             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18626         }
18627
18628
18629         var topSpace = b.y - c.y;
18630         var leftSpace = b.x - c.x;
18631
18632         this.resetConstraints();
18633         this.setXConstraint(leftSpace - (pad.left||0), // left
18634                 c.width - leftSpace - b.width - (pad.right||0) //right
18635         );
18636         this.setYConstraint(topSpace - (pad.top||0), //top
18637                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18638         );
18639     },
18640
18641     /**
18642      * Returns a reference to the linked element
18643      * @method getEl
18644      * @return {HTMLElement} the html element
18645      */
18646     getEl: function() {
18647         if (!this._domRef) {
18648             this._domRef = Roo.getDom(this.id);
18649         }
18650
18651         return this._domRef;
18652     },
18653
18654     /**
18655      * Returns a reference to the actual element to drag.  By default this is
18656      * the same as the html element, but it can be assigned to another
18657      * element. An example of this can be found in Roo.dd.DDProxy
18658      * @method getDragEl
18659      * @return {HTMLElement} the html element
18660      */
18661     getDragEl: function() {
18662         return Roo.getDom(this.dragElId);
18663     },
18664
18665     /**
18666      * Sets up the DragDrop object.  Must be called in the constructor of any
18667      * Roo.dd.DragDrop subclass
18668      * @method init
18669      * @param id the id of the linked element
18670      * @param {String} sGroup the group of related items
18671      * @param {object} config configuration attributes
18672      */
18673     init: function(id, sGroup, config) {
18674         this.initTarget(id, sGroup, config);
18675         if (!Roo.isTouch) {
18676             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18677         }
18678         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18679         // Event.on(this.id, "selectstart", Event.preventDefault);
18680     },
18681
18682     /**
18683      * Initializes Targeting functionality only... the object does not
18684      * get a mousedown handler.
18685      * @method initTarget
18686      * @param id the id of the linked element
18687      * @param {String} sGroup the group of related items
18688      * @param {object} config configuration attributes
18689      */
18690     initTarget: function(id, sGroup, config) {
18691
18692         // configuration attributes
18693         this.config = config || {};
18694
18695         // create a local reference to the drag and drop manager
18696         this.DDM = Roo.dd.DDM;
18697         // initialize the groups array
18698         this.groups = {};
18699
18700         // assume that we have an element reference instead of an id if the
18701         // parameter is not a string
18702         if (typeof id !== "string") {
18703             id = Roo.id(id);
18704         }
18705
18706         // set the id
18707         this.id = id;
18708
18709         // add to an interaction group
18710         this.addToGroup((sGroup) ? sGroup : "default");
18711
18712         // We don't want to register this as the handle with the manager
18713         // so we just set the id rather than calling the setter.
18714         this.handleElId = id;
18715
18716         // the linked element is the element that gets dragged by default
18717         this.setDragElId(id);
18718
18719         // by default, clicked anchors will not start drag operations.
18720         this.invalidHandleTypes = { A: "A" };
18721         this.invalidHandleIds = {};
18722         this.invalidHandleClasses = [];
18723
18724         this.applyConfig();
18725
18726         this.handleOnAvailable();
18727     },
18728
18729     /**
18730      * Applies the configuration parameters that were passed into the constructor.
18731      * This is supposed to happen at each level through the inheritance chain.  So
18732      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18733      * DragDrop in order to get all of the parameters that are available in
18734      * each object.
18735      * @method applyConfig
18736      */
18737     applyConfig: function() {
18738
18739         // configurable properties:
18740         //    padding, isTarget, maintainOffset, primaryButtonOnly
18741         this.padding           = this.config.padding || [0, 0, 0, 0];
18742         this.isTarget          = (this.config.isTarget !== false);
18743         this.maintainOffset    = (this.config.maintainOffset);
18744         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18745
18746     },
18747
18748     /**
18749      * Executed when the linked element is available
18750      * @method handleOnAvailable
18751      * @private
18752      */
18753     handleOnAvailable: function() {
18754         this.available = true;
18755         this.resetConstraints();
18756         this.onAvailable();
18757     },
18758
18759      /**
18760      * Configures the padding for the target zone in px.  Effectively expands
18761      * (or reduces) the virtual object size for targeting calculations.
18762      * Supports css-style shorthand; if only one parameter is passed, all sides
18763      * will have that padding, and if only two are passed, the top and bottom
18764      * will have the first param, the left and right the second.
18765      * @method setPadding
18766      * @param {int} iTop    Top pad
18767      * @param {int} iRight  Right pad
18768      * @param {int} iBot    Bot pad
18769      * @param {int} iLeft   Left pad
18770      */
18771     setPadding: function(iTop, iRight, iBot, iLeft) {
18772         // this.padding = [iLeft, iRight, iTop, iBot];
18773         if (!iRight && 0 !== iRight) {
18774             this.padding = [iTop, iTop, iTop, iTop];
18775         } else if (!iBot && 0 !== iBot) {
18776             this.padding = [iTop, iRight, iTop, iRight];
18777         } else {
18778             this.padding = [iTop, iRight, iBot, iLeft];
18779         }
18780     },
18781
18782     /**
18783      * Stores the initial placement of the linked element.
18784      * @method setInitialPosition
18785      * @param {int} diffX   the X offset, default 0
18786      * @param {int} diffY   the Y offset, default 0
18787      */
18788     setInitPosition: function(diffX, diffY) {
18789         var el = this.getEl();
18790
18791         if (!this.DDM.verifyEl(el)) {
18792             return;
18793         }
18794
18795         var dx = diffX || 0;
18796         var dy = diffY || 0;
18797
18798         var p = Dom.getXY( el );
18799
18800         this.initPageX = p[0] - dx;
18801         this.initPageY = p[1] - dy;
18802
18803         this.lastPageX = p[0];
18804         this.lastPageY = p[1];
18805
18806
18807         this.setStartPosition(p);
18808     },
18809
18810     /**
18811      * Sets the start position of the element.  This is set when the obj
18812      * is initialized, the reset when a drag is started.
18813      * @method setStartPosition
18814      * @param pos current position (from previous lookup)
18815      * @private
18816      */
18817     setStartPosition: function(pos) {
18818         var p = pos || Dom.getXY( this.getEl() );
18819         this.deltaSetXY = null;
18820
18821         this.startPageX = p[0];
18822         this.startPageY = p[1];
18823     },
18824
18825     /**
18826      * Add this instance to a group of related drag/drop objects.  All
18827      * instances belong to at least one group, and can belong to as many
18828      * groups as needed.
18829      * @method addToGroup
18830      * @param sGroup {string} the name of the group
18831      */
18832     addToGroup: function(sGroup) {
18833         this.groups[sGroup] = true;
18834         this.DDM.regDragDrop(this, sGroup);
18835     },
18836
18837     /**
18838      * Remove's this instance from the supplied interaction group
18839      * @method removeFromGroup
18840      * @param {string}  sGroup  The group to drop
18841      */
18842     removeFromGroup: function(sGroup) {
18843         if (this.groups[sGroup]) {
18844             delete this.groups[sGroup];
18845         }
18846
18847         this.DDM.removeDDFromGroup(this, sGroup);
18848     },
18849
18850     /**
18851      * Allows you to specify that an element other than the linked element
18852      * will be moved with the cursor during a drag
18853      * @method setDragElId
18854      * @param id {string} the id of the element that will be used to initiate the drag
18855      */
18856     setDragElId: function(id) {
18857         this.dragElId = id;
18858     },
18859
18860     /**
18861      * Allows you to specify a child of the linked element that should be
18862      * used to initiate the drag operation.  An example of this would be if
18863      * you have a content div with text and links.  Clicking anywhere in the
18864      * content area would normally start the drag operation.  Use this method
18865      * to specify that an element inside of the content div is the element
18866      * that starts the drag operation.
18867      * @method setHandleElId
18868      * @param id {string} the id of the element that will be used to
18869      * initiate the drag.
18870      */
18871     setHandleElId: function(id) {
18872         if (typeof id !== "string") {
18873             id = Roo.id(id);
18874         }
18875         this.handleElId = id;
18876         this.DDM.regHandle(this.id, id);
18877     },
18878
18879     /**
18880      * Allows you to set an element outside of the linked element as a drag
18881      * handle
18882      * @method setOuterHandleElId
18883      * @param id the id of the element that will be used to initiate the drag
18884      */
18885     setOuterHandleElId: function(id) {
18886         if (typeof id !== "string") {
18887             id = Roo.id(id);
18888         }
18889         Event.on(id, "mousedown",
18890                 this.handleMouseDown, this);
18891         this.setHandleElId(id);
18892
18893         this.hasOuterHandles = true;
18894     },
18895
18896     /**
18897      * Remove all drag and drop hooks for this element
18898      * @method unreg
18899      */
18900     unreg: function() {
18901         Event.un(this.id, "mousedown",
18902                 this.handleMouseDown);
18903         Event.un(this.id, "touchstart",
18904                 this.handleMouseDown);
18905         this._domRef = null;
18906         this.DDM._remove(this);
18907     },
18908
18909     destroy : function(){
18910         this.unreg();
18911     },
18912
18913     /**
18914      * Returns true if this instance is locked, or the drag drop mgr is locked
18915      * (meaning that all drag/drop is disabled on the page.)
18916      * @method isLocked
18917      * @return {boolean} true if this obj or all drag/drop is locked, else
18918      * false
18919      */
18920     isLocked: function() {
18921         return (this.DDM.isLocked() || this.locked);
18922     },
18923
18924     /**
18925      * Fired when this object is clicked
18926      * @method handleMouseDown
18927      * @param {Event} e
18928      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18929      * @private
18930      */
18931     handleMouseDown: function(e, oDD){
18932      
18933         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18934             //Roo.log('not touch/ button !=0');
18935             return;
18936         }
18937         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18938             return; // double touch..
18939         }
18940         
18941
18942         if (this.isLocked()) {
18943             //Roo.log('locked');
18944             return;
18945         }
18946
18947         this.DDM.refreshCache(this.groups);
18948 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18949         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18950         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18951             //Roo.log('no outer handes or not over target');
18952                 // do nothing.
18953         } else {
18954 //            Roo.log('check validator');
18955             if (this.clickValidator(e)) {
18956 //                Roo.log('validate success');
18957                 // set the initial element position
18958                 this.setStartPosition();
18959
18960
18961                 this.b4MouseDown(e);
18962                 this.onMouseDown(e);
18963
18964                 this.DDM.handleMouseDown(e, this);
18965
18966                 this.DDM.stopEvent(e);
18967             } else {
18968
18969
18970             }
18971         }
18972     },
18973
18974     clickValidator: function(e) {
18975         var target = e.getTarget();
18976         return ( this.isValidHandleChild(target) &&
18977                     (this.id == this.handleElId ||
18978                         this.DDM.handleWasClicked(target, this.id)) );
18979     },
18980
18981     /**
18982      * Allows you to specify a tag name that should not start a drag operation
18983      * when clicked.  This is designed to facilitate embedding links within a
18984      * drag handle that do something other than start the drag.
18985      * @method addInvalidHandleType
18986      * @param {string} tagName the type of element to exclude
18987      */
18988     addInvalidHandleType: function(tagName) {
18989         var type = tagName.toUpperCase();
18990         this.invalidHandleTypes[type] = type;
18991     },
18992
18993     /**
18994      * Lets you to specify an element id for a child of a drag handle
18995      * that should not initiate a drag
18996      * @method addInvalidHandleId
18997      * @param {string} id the element id of the element you wish to ignore
18998      */
18999     addInvalidHandleId: function(id) {
19000         if (typeof id !== "string") {
19001             id = Roo.id(id);
19002         }
19003         this.invalidHandleIds[id] = id;
19004     },
19005
19006     /**
19007      * Lets you specify a css class of elements that will not initiate a drag
19008      * @method addInvalidHandleClass
19009      * @param {string} cssClass the class of the elements you wish to ignore
19010      */
19011     addInvalidHandleClass: function(cssClass) {
19012         this.invalidHandleClasses.push(cssClass);
19013     },
19014
19015     /**
19016      * Unsets an excluded tag name set by addInvalidHandleType
19017      * @method removeInvalidHandleType
19018      * @param {string} tagName the type of element to unexclude
19019      */
19020     removeInvalidHandleType: function(tagName) {
19021         var type = tagName.toUpperCase();
19022         // this.invalidHandleTypes[type] = null;
19023         delete this.invalidHandleTypes[type];
19024     },
19025
19026     /**
19027      * Unsets an invalid handle id
19028      * @method removeInvalidHandleId
19029      * @param {string} id the id of the element to re-enable
19030      */
19031     removeInvalidHandleId: function(id) {
19032         if (typeof id !== "string") {
19033             id = Roo.id(id);
19034         }
19035         delete this.invalidHandleIds[id];
19036     },
19037
19038     /**
19039      * Unsets an invalid css class
19040      * @method removeInvalidHandleClass
19041      * @param {string} cssClass the class of the element(s) you wish to
19042      * re-enable
19043      */
19044     removeInvalidHandleClass: function(cssClass) {
19045         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19046             if (this.invalidHandleClasses[i] == cssClass) {
19047                 delete this.invalidHandleClasses[i];
19048             }
19049         }
19050     },
19051
19052     /**
19053      * Checks the tag exclusion list to see if this click should be ignored
19054      * @method isValidHandleChild
19055      * @param {HTMLElement} node the HTMLElement to evaluate
19056      * @return {boolean} true if this is a valid tag type, false if not
19057      */
19058     isValidHandleChild: function(node) {
19059
19060         var valid = true;
19061         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19062         var nodeName;
19063         try {
19064             nodeName = node.nodeName.toUpperCase();
19065         } catch(e) {
19066             nodeName = node.nodeName;
19067         }
19068         valid = valid && !this.invalidHandleTypes[nodeName];
19069         valid = valid && !this.invalidHandleIds[node.id];
19070
19071         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19072             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19073         }
19074
19075
19076         return valid;
19077
19078     },
19079
19080     /**
19081      * Create the array of horizontal tick marks if an interval was specified
19082      * in setXConstraint().
19083      * @method setXTicks
19084      * @private
19085      */
19086     setXTicks: function(iStartX, iTickSize) {
19087         this.xTicks = [];
19088         this.xTickSize = iTickSize;
19089
19090         var tickMap = {};
19091
19092         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19093             if (!tickMap[i]) {
19094                 this.xTicks[this.xTicks.length] = i;
19095                 tickMap[i] = true;
19096             }
19097         }
19098
19099         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19100             if (!tickMap[i]) {
19101                 this.xTicks[this.xTicks.length] = i;
19102                 tickMap[i] = true;
19103             }
19104         }
19105
19106         this.xTicks.sort(this.DDM.numericSort) ;
19107     },
19108
19109     /**
19110      * Create the array of vertical tick marks if an interval was specified in
19111      * setYConstraint().
19112      * @method setYTicks
19113      * @private
19114      */
19115     setYTicks: function(iStartY, iTickSize) {
19116         this.yTicks = [];
19117         this.yTickSize = iTickSize;
19118
19119         var tickMap = {};
19120
19121         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19122             if (!tickMap[i]) {
19123                 this.yTicks[this.yTicks.length] = i;
19124                 tickMap[i] = true;
19125             }
19126         }
19127
19128         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19129             if (!tickMap[i]) {
19130                 this.yTicks[this.yTicks.length] = i;
19131                 tickMap[i] = true;
19132             }
19133         }
19134
19135         this.yTicks.sort(this.DDM.numericSort) ;
19136     },
19137
19138     /**
19139      * By default, the element can be dragged any place on the screen.  Use
19140      * this method to limit the horizontal travel of the element.  Pass in
19141      * 0,0 for the parameters if you want to lock the drag to the y axis.
19142      * @method setXConstraint
19143      * @param {int} iLeft the number of pixels the element can move to the left
19144      * @param {int} iRight the number of pixels the element can move to the
19145      * right
19146      * @param {int} iTickSize optional parameter for specifying that the
19147      * element
19148      * should move iTickSize pixels at a time.
19149      */
19150     setXConstraint: function(iLeft, iRight, iTickSize) {
19151         this.leftConstraint = iLeft;
19152         this.rightConstraint = iRight;
19153
19154         this.minX = this.initPageX - iLeft;
19155         this.maxX = this.initPageX + iRight;
19156         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19157
19158         this.constrainX = true;
19159     },
19160
19161     /**
19162      * Clears any constraints applied to this instance.  Also clears ticks
19163      * since they can't exist independent of a constraint at this time.
19164      * @method clearConstraints
19165      */
19166     clearConstraints: function() {
19167         this.constrainX = false;
19168         this.constrainY = false;
19169         this.clearTicks();
19170     },
19171
19172     /**
19173      * Clears any tick interval defined for this instance
19174      * @method clearTicks
19175      */
19176     clearTicks: function() {
19177         this.xTicks = null;
19178         this.yTicks = null;
19179         this.xTickSize = 0;
19180         this.yTickSize = 0;
19181     },
19182
19183     /**
19184      * By default, the element can be dragged any place on the screen.  Set
19185      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19186      * parameters if you want to lock the drag to the x axis.
19187      * @method setYConstraint
19188      * @param {int} iUp the number of pixels the element can move up
19189      * @param {int} iDown the number of pixels the element can move down
19190      * @param {int} iTickSize optional parameter for specifying that the
19191      * element should move iTickSize pixels at a time.
19192      */
19193     setYConstraint: function(iUp, iDown, iTickSize) {
19194         this.topConstraint = iUp;
19195         this.bottomConstraint = iDown;
19196
19197         this.minY = this.initPageY - iUp;
19198         this.maxY = this.initPageY + iDown;
19199         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19200
19201         this.constrainY = true;
19202
19203     },
19204
19205     /**
19206      * resetConstraints must be called if you manually reposition a dd element.
19207      * @method resetConstraints
19208      * @param {boolean} maintainOffset
19209      */
19210     resetConstraints: function() {
19211
19212
19213         // Maintain offsets if necessary
19214         if (this.initPageX || this.initPageX === 0) {
19215             // figure out how much this thing has moved
19216             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19217             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19218
19219             this.setInitPosition(dx, dy);
19220
19221         // This is the first time we have detected the element's position
19222         } else {
19223             this.setInitPosition();
19224         }
19225
19226         if (this.constrainX) {
19227             this.setXConstraint( this.leftConstraint,
19228                                  this.rightConstraint,
19229                                  this.xTickSize        );
19230         }
19231
19232         if (this.constrainY) {
19233             this.setYConstraint( this.topConstraint,
19234                                  this.bottomConstraint,
19235                                  this.yTickSize         );
19236         }
19237     },
19238
19239     /**
19240      * Normally the drag element is moved pixel by pixel, but we can specify
19241      * that it move a number of pixels at a time.  This method resolves the
19242      * location when we have it set up like this.
19243      * @method getTick
19244      * @param {int} val where we want to place the object
19245      * @param {int[]} tickArray sorted array of valid points
19246      * @return {int} the closest tick
19247      * @private
19248      */
19249     getTick: function(val, tickArray) {
19250
19251         if (!tickArray) {
19252             // If tick interval is not defined, it is effectively 1 pixel,
19253             // so we return the value passed to us.
19254             return val;
19255         } else if (tickArray[0] >= val) {
19256             // The value is lower than the first tick, so we return the first
19257             // tick.
19258             return tickArray[0];
19259         } else {
19260             for (var i=0, len=tickArray.length; i<len; ++i) {
19261                 var next = i + 1;
19262                 if (tickArray[next] && tickArray[next] >= val) {
19263                     var diff1 = val - tickArray[i];
19264                     var diff2 = tickArray[next] - val;
19265                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19266                 }
19267             }
19268
19269             // The value is larger than the last tick, so we return the last
19270             // tick.
19271             return tickArray[tickArray.length - 1];
19272         }
19273     },
19274
19275     /**
19276      * toString method
19277      * @method toString
19278      * @return {string} string representation of the dd obj
19279      */
19280     toString: function() {
19281         return ("DragDrop " + this.id);
19282     }
19283
19284 });
19285
19286 })();
19287 /*
19288  * Based on:
19289  * Ext JS Library 1.1.1
19290  * Copyright(c) 2006-2007, Ext JS, LLC.
19291  *
19292  * Originally Released Under LGPL - original licence link has changed is not relivant.
19293  *
19294  * Fork - LGPL
19295  * <script type="text/javascript">
19296  */
19297
19298
19299 /**
19300  * The drag and drop utility provides a framework for building drag and drop
19301  * applications.  In addition to enabling drag and drop for specific elements,
19302  * the drag and drop elements are tracked by the manager class, and the
19303  * interactions between the various elements are tracked during the drag and
19304  * the implementing code is notified about these important moments.
19305  */
19306
19307 // Only load the library once.  Rewriting the manager class would orphan
19308 // existing drag and drop instances.
19309 if (!Roo.dd.DragDropMgr) {
19310
19311 /**
19312  * @class Roo.dd.DragDropMgr
19313  * DragDropMgr is a singleton that tracks the element interaction for
19314  * all DragDrop items in the window.  Generally, you will not call
19315  * this class directly, but it does have helper methods that could
19316  * be useful in your DragDrop implementations.
19317  * @singleton
19318  */
19319 Roo.dd.DragDropMgr = function() {
19320
19321     var Event = Roo.EventManager;
19322
19323     return {
19324
19325         /**
19326          * Two dimensional Array of registered DragDrop objects.  The first
19327          * dimension is the DragDrop item group, the second the DragDrop
19328          * object.
19329          * @property ids
19330          * @type {string: string}
19331          * @private
19332          * @static
19333          */
19334         ids: {},
19335
19336         /**
19337          * Array of element ids defined as drag handles.  Used to determine
19338          * if the element that generated the mousedown event is actually the
19339          * handle and not the html element itself.
19340          * @property handleIds
19341          * @type {string: string}
19342          * @private
19343          * @static
19344          */
19345         handleIds: {},
19346
19347         /**
19348          * the DragDrop object that is currently being dragged
19349          * @property dragCurrent
19350          * @type DragDrop
19351          * @private
19352          * @static
19353          **/
19354         dragCurrent: null,
19355
19356         /**
19357          * the DragDrop object(s) that are being hovered over
19358          * @property dragOvers
19359          * @type Array
19360          * @private
19361          * @static
19362          */
19363         dragOvers: {},
19364
19365         /**
19366          * the X distance between the cursor and the object being dragged
19367          * @property deltaX
19368          * @type int
19369          * @private
19370          * @static
19371          */
19372         deltaX: 0,
19373
19374         /**
19375          * the Y distance between the cursor and the object being dragged
19376          * @property deltaY
19377          * @type int
19378          * @private
19379          * @static
19380          */
19381         deltaY: 0,
19382
19383         /**
19384          * Flag to determine if we should prevent the default behavior of the
19385          * events we define. By default this is true, but this can be set to
19386          * false if you need the default behavior (not recommended)
19387          * @property preventDefault
19388          * @type boolean
19389          * @static
19390          */
19391         preventDefault: true,
19392
19393         /**
19394          * Flag to determine if we should stop the propagation of the events
19395          * we generate. This is true by default but you may want to set it to
19396          * false if the html element contains other features that require the
19397          * mouse click.
19398          * @property stopPropagation
19399          * @type boolean
19400          * @static
19401          */
19402         stopPropagation: true,
19403
19404         /**
19405          * Internal flag that is set to true when drag and drop has been
19406          * intialized
19407          * @property initialized
19408          * @private
19409          * @static
19410          */
19411         initalized: false,
19412
19413         /**
19414          * All drag and drop can be disabled.
19415          * @property locked
19416          * @private
19417          * @static
19418          */
19419         locked: false,
19420
19421         /**
19422          * Called the first time an element is registered.
19423          * @method init
19424          * @private
19425          * @static
19426          */
19427         init: function() {
19428             this.initialized = true;
19429         },
19430
19431         /**
19432          * In point mode, drag and drop interaction is defined by the
19433          * location of the cursor during the drag/drop
19434          * @property POINT
19435          * @type int
19436          * @static
19437          */
19438         POINT: 0,
19439
19440         /**
19441          * In intersect mode, drag and drop interactio nis defined by the
19442          * overlap of two or more drag and drop objects.
19443          * @property INTERSECT
19444          * @type int
19445          * @static
19446          */
19447         INTERSECT: 1,
19448
19449         /**
19450          * The current drag and drop mode.  Default: POINT
19451          * @property mode
19452          * @type int
19453          * @static
19454          */
19455         mode: 0,
19456
19457         /**
19458          * Runs method on all drag and drop objects
19459          * @method _execOnAll
19460          * @private
19461          * @static
19462          */
19463         _execOnAll: function(sMethod, args) {
19464             for (var i in this.ids) {
19465                 for (var j in this.ids[i]) {
19466                     var oDD = this.ids[i][j];
19467                     if (! this.isTypeOfDD(oDD)) {
19468                         continue;
19469                     }
19470                     oDD[sMethod].apply(oDD, args);
19471                 }
19472             }
19473         },
19474
19475         /**
19476          * Drag and drop initialization.  Sets up the global event handlers
19477          * @method _onLoad
19478          * @private
19479          * @static
19480          */
19481         _onLoad: function() {
19482
19483             this.init();
19484
19485             if (!Roo.isTouch) {
19486                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19487                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19488             }
19489             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19490             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19491             
19492             Event.on(window,   "unload",    this._onUnload, this, true);
19493             Event.on(window,   "resize",    this._onResize, this, true);
19494             // Event.on(window,   "mouseout",    this._test);
19495
19496         },
19497
19498         /**
19499          * Reset constraints on all drag and drop objs
19500          * @method _onResize
19501          * @private
19502          * @static
19503          */
19504         _onResize: function(e) {
19505             this._execOnAll("resetConstraints", []);
19506         },
19507
19508         /**
19509          * Lock all drag and drop functionality
19510          * @method lock
19511          * @static
19512          */
19513         lock: function() { this.locked = true; },
19514
19515         /**
19516          * Unlock all drag and drop functionality
19517          * @method unlock
19518          * @static
19519          */
19520         unlock: function() { this.locked = false; },
19521
19522         /**
19523          * Is drag and drop locked?
19524          * @method isLocked
19525          * @return {boolean} True if drag and drop is locked, false otherwise.
19526          * @static
19527          */
19528         isLocked: function() { return this.locked; },
19529
19530         /**
19531          * Location cache that is set for all drag drop objects when a drag is
19532          * initiated, cleared when the drag is finished.
19533          * @property locationCache
19534          * @private
19535          * @static
19536          */
19537         locationCache: {},
19538
19539         /**
19540          * Set useCache to false if you want to force object the lookup of each
19541          * drag and drop linked element constantly during a drag.
19542          * @property useCache
19543          * @type boolean
19544          * @static
19545          */
19546         useCache: true,
19547
19548         /**
19549          * The number of pixels that the mouse needs to move after the
19550          * mousedown before the drag is initiated.  Default=3;
19551          * @property clickPixelThresh
19552          * @type int
19553          * @static
19554          */
19555         clickPixelThresh: 3,
19556
19557         /**
19558          * The number of milliseconds after the mousedown event to initiate the
19559          * drag if we don't get a mouseup event. Default=1000
19560          * @property clickTimeThresh
19561          * @type int
19562          * @static
19563          */
19564         clickTimeThresh: 350,
19565
19566         /**
19567          * Flag that indicates that either the drag pixel threshold or the
19568          * mousdown time threshold has been met
19569          * @property dragThreshMet
19570          * @type boolean
19571          * @private
19572          * @static
19573          */
19574         dragThreshMet: false,
19575
19576         /**
19577          * Timeout used for the click time threshold
19578          * @property clickTimeout
19579          * @type Object
19580          * @private
19581          * @static
19582          */
19583         clickTimeout: null,
19584
19585         /**
19586          * The X position of the mousedown event stored for later use when a
19587          * drag threshold is met.
19588          * @property startX
19589          * @type int
19590          * @private
19591          * @static
19592          */
19593         startX: 0,
19594
19595         /**
19596          * The Y position of the mousedown event stored for later use when a
19597          * drag threshold is met.
19598          * @property startY
19599          * @type int
19600          * @private
19601          * @static
19602          */
19603         startY: 0,
19604
19605         /**
19606          * Each DragDrop instance must be registered with the DragDropMgr.
19607          * This is executed in DragDrop.init()
19608          * @method regDragDrop
19609          * @param {DragDrop} oDD the DragDrop object to register
19610          * @param {String} sGroup the name of the group this element belongs to
19611          * @static
19612          */
19613         regDragDrop: function(oDD, sGroup) {
19614             if (!this.initialized) { this.init(); }
19615
19616             if (!this.ids[sGroup]) {
19617                 this.ids[sGroup] = {};
19618             }
19619             this.ids[sGroup][oDD.id] = oDD;
19620         },
19621
19622         /**
19623          * Removes the supplied dd instance from the supplied group. Executed
19624          * by DragDrop.removeFromGroup, so don't call this function directly.
19625          * @method removeDDFromGroup
19626          * @private
19627          * @static
19628          */
19629         removeDDFromGroup: function(oDD, sGroup) {
19630             if (!this.ids[sGroup]) {
19631                 this.ids[sGroup] = {};
19632             }
19633
19634             var obj = this.ids[sGroup];
19635             if (obj && obj[oDD.id]) {
19636                 delete obj[oDD.id];
19637             }
19638         },
19639
19640         /**
19641          * Unregisters a drag and drop item.  This is executed in
19642          * DragDrop.unreg, use that method instead of calling this directly.
19643          * @method _remove
19644          * @private
19645          * @static
19646          */
19647         _remove: function(oDD) {
19648             for (var g in oDD.groups) {
19649                 if (g && this.ids[g][oDD.id]) {
19650                     delete this.ids[g][oDD.id];
19651                 }
19652             }
19653             delete this.handleIds[oDD.id];
19654         },
19655
19656         /**
19657          * Each DragDrop handle element must be registered.  This is done
19658          * automatically when executing DragDrop.setHandleElId()
19659          * @method regHandle
19660          * @param {String} sDDId the DragDrop id this element is a handle for
19661          * @param {String} sHandleId the id of the element that is the drag
19662          * handle
19663          * @static
19664          */
19665         regHandle: function(sDDId, sHandleId) {
19666             if (!this.handleIds[sDDId]) {
19667                 this.handleIds[sDDId] = {};
19668             }
19669             this.handleIds[sDDId][sHandleId] = sHandleId;
19670         },
19671
19672         /**
19673          * Utility function to determine if a given element has been
19674          * registered as a drag drop item.
19675          * @method isDragDrop
19676          * @param {String} id the element id to check
19677          * @return {boolean} true if this element is a DragDrop item,
19678          * false otherwise
19679          * @static
19680          */
19681         isDragDrop: function(id) {
19682             return ( this.getDDById(id) ) ? true : false;
19683         },
19684
19685         /**
19686          * Returns the drag and drop instances that are in all groups the
19687          * passed in instance belongs to.
19688          * @method getRelated
19689          * @param {DragDrop} p_oDD the obj to get related data for
19690          * @param {boolean} bTargetsOnly if true, only return targetable objs
19691          * @return {DragDrop[]} the related instances
19692          * @static
19693          */
19694         getRelated: function(p_oDD, bTargetsOnly) {
19695             var oDDs = [];
19696             for (var i in p_oDD.groups) {
19697                 for (j in this.ids[i]) {
19698                     var dd = this.ids[i][j];
19699                     if (! this.isTypeOfDD(dd)) {
19700                         continue;
19701                     }
19702                     if (!bTargetsOnly || dd.isTarget) {
19703                         oDDs[oDDs.length] = dd;
19704                     }
19705                 }
19706             }
19707
19708             return oDDs;
19709         },
19710
19711         /**
19712          * Returns true if the specified dd target is a legal target for
19713          * the specifice drag obj
19714          * @method isLegalTarget
19715          * @param {DragDrop} the drag obj
19716          * @param {DragDrop} the target
19717          * @return {boolean} true if the target is a legal target for the
19718          * dd obj
19719          * @static
19720          */
19721         isLegalTarget: function (oDD, oTargetDD) {
19722             var targets = this.getRelated(oDD, true);
19723             for (var i=0, len=targets.length;i<len;++i) {
19724                 if (targets[i].id == oTargetDD.id) {
19725                     return true;
19726                 }
19727             }
19728
19729             return false;
19730         },
19731
19732         /**
19733          * My goal is to be able to transparently determine if an object is
19734          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19735          * returns "object", oDD.constructor.toString() always returns
19736          * "DragDrop" and not the name of the subclass.  So for now it just
19737          * evaluates a well-known variable in DragDrop.
19738          * @method isTypeOfDD
19739          * @param {Object} the object to evaluate
19740          * @return {boolean} true if typeof oDD = DragDrop
19741          * @static
19742          */
19743         isTypeOfDD: function (oDD) {
19744             return (oDD && oDD.__ygDragDrop);
19745         },
19746
19747         /**
19748          * Utility function to determine if a given element has been
19749          * registered as a drag drop handle for the given Drag Drop object.
19750          * @method isHandle
19751          * @param {String} id the element id to check
19752          * @return {boolean} true if this element is a DragDrop handle, false
19753          * otherwise
19754          * @static
19755          */
19756         isHandle: function(sDDId, sHandleId) {
19757             return ( this.handleIds[sDDId] &&
19758                             this.handleIds[sDDId][sHandleId] );
19759         },
19760
19761         /**
19762          * Returns the DragDrop instance for a given id
19763          * @method getDDById
19764          * @param {String} id the id of the DragDrop object
19765          * @return {DragDrop} the drag drop object, null if it is not found
19766          * @static
19767          */
19768         getDDById: function(id) {
19769             for (var i in this.ids) {
19770                 if (this.ids[i][id]) {
19771                     return this.ids[i][id];
19772                 }
19773             }
19774             return null;
19775         },
19776
19777         /**
19778          * Fired after a registered DragDrop object gets the mousedown event.
19779          * Sets up the events required to track the object being dragged
19780          * @method handleMouseDown
19781          * @param {Event} e the event
19782          * @param oDD the DragDrop object being dragged
19783          * @private
19784          * @static
19785          */
19786         handleMouseDown: function(e, oDD) {
19787             if(Roo.QuickTips){
19788                 Roo.QuickTips.disable();
19789             }
19790             this.currentTarget = e.getTarget();
19791
19792             this.dragCurrent = oDD;
19793
19794             var el = oDD.getEl();
19795
19796             // track start position
19797             this.startX = e.getPageX();
19798             this.startY = e.getPageY();
19799
19800             this.deltaX = this.startX - el.offsetLeft;
19801             this.deltaY = this.startY - el.offsetTop;
19802
19803             this.dragThreshMet = false;
19804
19805             this.clickTimeout = setTimeout(
19806                     function() {
19807                         var DDM = Roo.dd.DDM;
19808                         DDM.startDrag(DDM.startX, DDM.startY);
19809                     },
19810                     this.clickTimeThresh );
19811         },
19812
19813         /**
19814          * Fired when either the drag pixel threshol or the mousedown hold
19815          * time threshold has been met.
19816          * @method startDrag
19817          * @param x {int} the X position of the original mousedown
19818          * @param y {int} the Y position of the original mousedown
19819          * @static
19820          */
19821         startDrag: function(x, y) {
19822             clearTimeout(this.clickTimeout);
19823             if (this.dragCurrent) {
19824                 this.dragCurrent.b4StartDrag(x, y);
19825                 this.dragCurrent.startDrag(x, y);
19826             }
19827             this.dragThreshMet = true;
19828         },
19829
19830         /**
19831          * Internal function to handle the mouseup event.  Will be invoked
19832          * from the context of the document.
19833          * @method handleMouseUp
19834          * @param {Event} e the event
19835          * @private
19836          * @static
19837          */
19838         handleMouseUp: function(e) {
19839
19840             if(Roo.QuickTips){
19841                 Roo.QuickTips.enable();
19842             }
19843             if (! this.dragCurrent) {
19844                 return;
19845             }
19846
19847             clearTimeout(this.clickTimeout);
19848
19849             if (this.dragThreshMet) {
19850                 this.fireEvents(e, true);
19851             } else {
19852             }
19853
19854             this.stopDrag(e);
19855
19856             this.stopEvent(e);
19857         },
19858
19859         /**
19860          * Utility to stop event propagation and event default, if these
19861          * features are turned on.
19862          * @method stopEvent
19863          * @param {Event} e the event as returned by this.getEvent()
19864          * @static
19865          */
19866         stopEvent: function(e){
19867             if(this.stopPropagation) {
19868                 e.stopPropagation();
19869             }
19870
19871             if (this.preventDefault) {
19872                 e.preventDefault();
19873             }
19874         },
19875
19876         /**
19877          * Internal function to clean up event handlers after the drag
19878          * operation is complete
19879          * @method stopDrag
19880          * @param {Event} e the event
19881          * @private
19882          * @static
19883          */
19884         stopDrag: function(e) {
19885             // Fire the drag end event for the item that was dragged
19886             if (this.dragCurrent) {
19887                 if (this.dragThreshMet) {
19888                     this.dragCurrent.b4EndDrag(e);
19889                     this.dragCurrent.endDrag(e);
19890                 }
19891
19892                 this.dragCurrent.onMouseUp(e);
19893             }
19894
19895             this.dragCurrent = null;
19896             this.dragOvers = {};
19897         },
19898
19899         /**
19900          * Internal function to handle the mousemove event.  Will be invoked
19901          * from the context of the html element.
19902          *
19903          * @TODO figure out what we can do about mouse events lost when the
19904          * user drags objects beyond the window boundary.  Currently we can
19905          * detect this in internet explorer by verifying that the mouse is
19906          * down during the mousemove event.  Firefox doesn't give us the
19907          * button state on the mousemove event.
19908          * @method handleMouseMove
19909          * @param {Event} e the event
19910          * @private
19911          * @static
19912          */
19913         handleMouseMove: function(e) {
19914             if (! this.dragCurrent) {
19915                 return true;
19916             }
19917
19918             // var button = e.which || e.button;
19919
19920             // check for IE mouseup outside of page boundary
19921             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19922                 this.stopEvent(e);
19923                 return this.handleMouseUp(e);
19924             }
19925
19926             if (!this.dragThreshMet) {
19927                 var diffX = Math.abs(this.startX - e.getPageX());
19928                 var diffY = Math.abs(this.startY - e.getPageY());
19929                 if (diffX > this.clickPixelThresh ||
19930                             diffY > this.clickPixelThresh) {
19931                     this.startDrag(this.startX, this.startY);
19932                 }
19933             }
19934
19935             if (this.dragThreshMet) {
19936                 this.dragCurrent.b4Drag(e);
19937                 this.dragCurrent.onDrag(e);
19938                 if(!this.dragCurrent.moveOnly){
19939                     this.fireEvents(e, false);
19940                 }
19941             }
19942
19943             this.stopEvent(e);
19944
19945             return true;
19946         },
19947
19948         /**
19949          * Iterates over all of the DragDrop elements to find ones we are
19950          * hovering over or dropping on
19951          * @method fireEvents
19952          * @param {Event} e the event
19953          * @param {boolean} isDrop is this a drop op or a mouseover op?
19954          * @private
19955          * @static
19956          */
19957         fireEvents: function(e, isDrop) {
19958             var dc = this.dragCurrent;
19959
19960             // If the user did the mouse up outside of the window, we could
19961             // get here even though we have ended the drag.
19962             if (!dc || dc.isLocked()) {
19963                 return;
19964             }
19965
19966             var pt = e.getPoint();
19967
19968             // cache the previous dragOver array
19969             var oldOvers = [];
19970
19971             var outEvts   = [];
19972             var overEvts  = [];
19973             var dropEvts  = [];
19974             var enterEvts = [];
19975
19976             // Check to see if the object(s) we were hovering over is no longer
19977             // being hovered over so we can fire the onDragOut event
19978             for (var i in this.dragOvers) {
19979
19980                 var ddo = this.dragOvers[i];
19981
19982                 if (! this.isTypeOfDD(ddo)) {
19983                     continue;
19984                 }
19985
19986                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19987                     outEvts.push( ddo );
19988                 }
19989
19990                 oldOvers[i] = true;
19991                 delete this.dragOvers[i];
19992             }
19993
19994             for (var sGroup in dc.groups) {
19995
19996                 if ("string" != typeof sGroup) {
19997                     continue;
19998                 }
19999
20000                 for (i in this.ids[sGroup]) {
20001                     var oDD = this.ids[sGroup][i];
20002                     if (! this.isTypeOfDD(oDD)) {
20003                         continue;
20004                     }
20005
20006                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20007                         if (this.isOverTarget(pt, oDD, this.mode)) {
20008                             // look for drop interactions
20009                             if (isDrop) {
20010                                 dropEvts.push( oDD );
20011                             // look for drag enter and drag over interactions
20012                             } else {
20013
20014                                 // initial drag over: dragEnter fires
20015                                 if (!oldOvers[oDD.id]) {
20016                                     enterEvts.push( oDD );
20017                                 // subsequent drag overs: dragOver fires
20018                                 } else {
20019                                     overEvts.push( oDD );
20020                                 }
20021
20022                                 this.dragOvers[oDD.id] = oDD;
20023                             }
20024                         }
20025                     }
20026                 }
20027             }
20028
20029             if (this.mode) {
20030                 if (outEvts.length) {
20031                     dc.b4DragOut(e, outEvts);
20032                     dc.onDragOut(e, outEvts);
20033                 }
20034
20035                 if (enterEvts.length) {
20036                     dc.onDragEnter(e, enterEvts);
20037                 }
20038
20039                 if (overEvts.length) {
20040                     dc.b4DragOver(e, overEvts);
20041                     dc.onDragOver(e, overEvts);
20042                 }
20043
20044                 if (dropEvts.length) {
20045                     dc.b4DragDrop(e, dropEvts);
20046                     dc.onDragDrop(e, dropEvts);
20047                 }
20048
20049             } else {
20050                 // fire dragout events
20051                 var len = 0;
20052                 for (i=0, len=outEvts.length; i<len; ++i) {
20053                     dc.b4DragOut(e, outEvts[i].id);
20054                     dc.onDragOut(e, outEvts[i].id);
20055                 }
20056
20057                 // fire enter events
20058                 for (i=0,len=enterEvts.length; i<len; ++i) {
20059                     // dc.b4DragEnter(e, oDD.id);
20060                     dc.onDragEnter(e, enterEvts[i].id);
20061                 }
20062
20063                 // fire over events
20064                 for (i=0,len=overEvts.length; i<len; ++i) {
20065                     dc.b4DragOver(e, overEvts[i].id);
20066                     dc.onDragOver(e, overEvts[i].id);
20067                 }
20068
20069                 // fire drop events
20070                 for (i=0, len=dropEvts.length; i<len; ++i) {
20071                     dc.b4DragDrop(e, dropEvts[i].id);
20072                     dc.onDragDrop(e, dropEvts[i].id);
20073                 }
20074
20075             }
20076
20077             // notify about a drop that did not find a target
20078             if (isDrop && !dropEvts.length) {
20079                 dc.onInvalidDrop(e);
20080             }
20081
20082         },
20083
20084         /**
20085          * Helper function for getting the best match from the list of drag
20086          * and drop objects returned by the drag and drop events when we are
20087          * in INTERSECT mode.  It returns either the first object that the
20088          * cursor is over, or the object that has the greatest overlap with
20089          * the dragged element.
20090          * @method getBestMatch
20091          * @param  {DragDrop[]} dds The array of drag and drop objects
20092          * targeted
20093          * @return {DragDrop}       The best single match
20094          * @static
20095          */
20096         getBestMatch: function(dds) {
20097             var winner = null;
20098             // Return null if the input is not what we expect
20099             //if (!dds || !dds.length || dds.length == 0) {
20100                // winner = null;
20101             // If there is only one item, it wins
20102             //} else if (dds.length == 1) {
20103
20104             var len = dds.length;
20105
20106             if (len == 1) {
20107                 winner = dds[0];
20108             } else {
20109                 // Loop through the targeted items
20110                 for (var i=0; i<len; ++i) {
20111                     var dd = dds[i];
20112                     // If the cursor is over the object, it wins.  If the
20113                     // cursor is over multiple matches, the first one we come
20114                     // to wins.
20115                     if (dd.cursorIsOver) {
20116                         winner = dd;
20117                         break;
20118                     // Otherwise the object with the most overlap wins
20119                     } else {
20120                         if (!winner ||
20121                             winner.overlap.getArea() < dd.overlap.getArea()) {
20122                             winner = dd;
20123                         }
20124                     }
20125                 }
20126             }
20127
20128             return winner;
20129         },
20130
20131         /**
20132          * Refreshes the cache of the top-left and bottom-right points of the
20133          * drag and drop objects in the specified group(s).  This is in the
20134          * format that is stored in the drag and drop instance, so typical
20135          * usage is:
20136          * <code>
20137          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20138          * </code>
20139          * Alternatively:
20140          * <code>
20141          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20142          * </code>
20143          * @TODO this really should be an indexed array.  Alternatively this
20144          * method could accept both.
20145          * @method refreshCache
20146          * @param {Object} groups an associative array of groups to refresh
20147          * @static
20148          */
20149         refreshCache: function(groups) {
20150             for (var sGroup in groups) {
20151                 if ("string" != typeof sGroup) {
20152                     continue;
20153                 }
20154                 for (var i in this.ids[sGroup]) {
20155                     var oDD = this.ids[sGroup][i];
20156
20157                     if (this.isTypeOfDD(oDD)) {
20158                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20159                         var loc = this.getLocation(oDD);
20160                         if (loc) {
20161                             this.locationCache[oDD.id] = loc;
20162                         } else {
20163                             delete this.locationCache[oDD.id];
20164                             // this will unregister the drag and drop object if
20165                             // the element is not in a usable state
20166                             // oDD.unreg();
20167                         }
20168                     }
20169                 }
20170             }
20171         },
20172
20173         /**
20174          * This checks to make sure an element exists and is in the DOM.  The
20175          * main purpose is to handle cases where innerHTML is used to remove
20176          * drag and drop objects from the DOM.  IE provides an 'unspecified
20177          * error' when trying to access the offsetParent of such an element
20178          * @method verifyEl
20179          * @param {HTMLElement} el the element to check
20180          * @return {boolean} true if the element looks usable
20181          * @static
20182          */
20183         verifyEl: function(el) {
20184             if (el) {
20185                 var parent;
20186                 if(Roo.isIE){
20187                     try{
20188                         parent = el.offsetParent;
20189                     }catch(e){}
20190                 }else{
20191                     parent = el.offsetParent;
20192                 }
20193                 if (parent) {
20194                     return true;
20195                 }
20196             }
20197
20198             return false;
20199         },
20200
20201         /**
20202          * Returns a Region object containing the drag and drop element's position
20203          * and size, including the padding configured for it
20204          * @method getLocation
20205          * @param {DragDrop} oDD the drag and drop object to get the
20206          *                       location for
20207          * @return {Roo.lib.Region} a Region object representing the total area
20208          *                             the element occupies, including any padding
20209          *                             the instance is configured for.
20210          * @static
20211          */
20212         getLocation: function(oDD) {
20213             if (! this.isTypeOfDD(oDD)) {
20214                 return null;
20215             }
20216
20217             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20218
20219             try {
20220                 pos= Roo.lib.Dom.getXY(el);
20221             } catch (e) { }
20222
20223             if (!pos) {
20224                 return null;
20225             }
20226
20227             x1 = pos[0];
20228             x2 = x1 + el.offsetWidth;
20229             y1 = pos[1];
20230             y2 = y1 + el.offsetHeight;
20231
20232             t = y1 - oDD.padding[0];
20233             r = x2 + oDD.padding[1];
20234             b = y2 + oDD.padding[2];
20235             l = x1 - oDD.padding[3];
20236
20237             return new Roo.lib.Region( t, r, b, l );
20238         },
20239
20240         /**
20241          * Checks the cursor location to see if it over the target
20242          * @method isOverTarget
20243          * @param {Roo.lib.Point} pt The point to evaluate
20244          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20245          * @return {boolean} true if the mouse is over the target
20246          * @private
20247          * @static
20248          */
20249         isOverTarget: function(pt, oTarget, intersect) {
20250             // use cache if available
20251             var loc = this.locationCache[oTarget.id];
20252             if (!loc || !this.useCache) {
20253                 loc = this.getLocation(oTarget);
20254                 this.locationCache[oTarget.id] = loc;
20255
20256             }
20257
20258             if (!loc) {
20259                 return false;
20260             }
20261
20262             oTarget.cursorIsOver = loc.contains( pt );
20263
20264             // DragDrop is using this as a sanity check for the initial mousedown
20265             // in this case we are done.  In POINT mode, if the drag obj has no
20266             // contraints, we are also done. Otherwise we need to evaluate the
20267             // location of the target as related to the actual location of the
20268             // dragged element.
20269             var dc = this.dragCurrent;
20270             if (!dc || !dc.getTargetCoord ||
20271                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20272                 return oTarget.cursorIsOver;
20273             }
20274
20275             oTarget.overlap = null;
20276
20277             // Get the current location of the drag element, this is the
20278             // location of the mouse event less the delta that represents
20279             // where the original mousedown happened on the element.  We
20280             // need to consider constraints and ticks as well.
20281             var pos = dc.getTargetCoord(pt.x, pt.y);
20282
20283             var el = dc.getDragEl();
20284             var curRegion = new Roo.lib.Region( pos.y,
20285                                                    pos.x + el.offsetWidth,
20286                                                    pos.y + el.offsetHeight,
20287                                                    pos.x );
20288
20289             var overlap = curRegion.intersect(loc);
20290
20291             if (overlap) {
20292                 oTarget.overlap = overlap;
20293                 return (intersect) ? true : oTarget.cursorIsOver;
20294             } else {
20295                 return false;
20296             }
20297         },
20298
20299         /**
20300          * unload event handler
20301          * @method _onUnload
20302          * @private
20303          * @static
20304          */
20305         _onUnload: function(e, me) {
20306             Roo.dd.DragDropMgr.unregAll();
20307         },
20308
20309         /**
20310          * Cleans up the drag and drop events and objects.
20311          * @method unregAll
20312          * @private
20313          * @static
20314          */
20315         unregAll: function() {
20316
20317             if (this.dragCurrent) {
20318                 this.stopDrag();
20319                 this.dragCurrent = null;
20320             }
20321
20322             this._execOnAll("unreg", []);
20323
20324             for (i in this.elementCache) {
20325                 delete this.elementCache[i];
20326             }
20327
20328             this.elementCache = {};
20329             this.ids = {};
20330         },
20331
20332         /**
20333          * A cache of DOM elements
20334          * @property elementCache
20335          * @private
20336          * @static
20337          */
20338         elementCache: {},
20339
20340         /**
20341          * Get the wrapper for the DOM element specified
20342          * @method getElWrapper
20343          * @param {String} id the id of the element to get
20344          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20345          * @private
20346          * @deprecated This wrapper isn't that useful
20347          * @static
20348          */
20349         getElWrapper: function(id) {
20350             var oWrapper = this.elementCache[id];
20351             if (!oWrapper || !oWrapper.el) {
20352                 oWrapper = this.elementCache[id] =
20353                     new this.ElementWrapper(Roo.getDom(id));
20354             }
20355             return oWrapper;
20356         },
20357
20358         /**
20359          * Returns the actual DOM element
20360          * @method getElement
20361          * @param {String} id the id of the elment to get
20362          * @return {Object} The element
20363          * @deprecated use Roo.getDom instead
20364          * @static
20365          */
20366         getElement: function(id) {
20367             return Roo.getDom(id);
20368         },
20369
20370         /**
20371          * Returns the style property for the DOM element (i.e.,
20372          * document.getElById(id).style)
20373          * @method getCss
20374          * @param {String} id the id of the elment to get
20375          * @return {Object} The style property of the element
20376          * @deprecated use Roo.getDom instead
20377          * @static
20378          */
20379         getCss: function(id) {
20380             var el = Roo.getDom(id);
20381             return (el) ? el.style : null;
20382         },
20383
20384         /**
20385          * Inner class for cached elements
20386          * @class DragDropMgr.ElementWrapper
20387          * @for DragDropMgr
20388          * @private
20389          * @deprecated
20390          */
20391         ElementWrapper: function(el) {
20392                 /**
20393                  * The element
20394                  * @property el
20395                  */
20396                 this.el = el || null;
20397                 /**
20398                  * The element id
20399                  * @property id
20400                  */
20401                 this.id = this.el && el.id;
20402                 /**
20403                  * A reference to the style property
20404                  * @property css
20405                  */
20406                 this.css = this.el && el.style;
20407             },
20408
20409         /**
20410          * Returns the X position of an html element
20411          * @method getPosX
20412          * @param el the element for which to get the position
20413          * @return {int} the X coordinate
20414          * @for DragDropMgr
20415          * @deprecated use Roo.lib.Dom.getX instead
20416          * @static
20417          */
20418         getPosX: function(el) {
20419             return Roo.lib.Dom.getX(el);
20420         },
20421
20422         /**
20423          * Returns the Y position of an html element
20424          * @method getPosY
20425          * @param el the element for which to get the position
20426          * @return {int} the Y coordinate
20427          * @deprecated use Roo.lib.Dom.getY instead
20428          * @static
20429          */
20430         getPosY: function(el) {
20431             return Roo.lib.Dom.getY(el);
20432         },
20433
20434         /**
20435          * Swap two nodes.  In IE, we use the native method, for others we
20436          * emulate the IE behavior
20437          * @method swapNode
20438          * @param n1 the first node to swap
20439          * @param n2 the other node to swap
20440          * @static
20441          */
20442         swapNode: function(n1, n2) {
20443             if (n1.swapNode) {
20444                 n1.swapNode(n2);
20445             } else {
20446                 var p = n2.parentNode;
20447                 var s = n2.nextSibling;
20448
20449                 if (s == n1) {
20450                     p.insertBefore(n1, n2);
20451                 } else if (n2 == n1.nextSibling) {
20452                     p.insertBefore(n2, n1);
20453                 } else {
20454                     n1.parentNode.replaceChild(n2, n1);
20455                     p.insertBefore(n1, s);
20456                 }
20457             }
20458         },
20459
20460         /**
20461          * Returns the current scroll position
20462          * @method getScroll
20463          * @private
20464          * @static
20465          */
20466         getScroll: function () {
20467             var t, l, dde=document.documentElement, db=document.body;
20468             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20469                 t = dde.scrollTop;
20470                 l = dde.scrollLeft;
20471             } else if (db) {
20472                 t = db.scrollTop;
20473                 l = db.scrollLeft;
20474             } else {
20475
20476             }
20477             return { top: t, left: l };
20478         },
20479
20480         /**
20481          * Returns the specified element style property
20482          * @method getStyle
20483          * @param {HTMLElement} el          the element
20484          * @param {string}      styleProp   the style property
20485          * @return {string} The value of the style property
20486          * @deprecated use Roo.lib.Dom.getStyle
20487          * @static
20488          */
20489         getStyle: function(el, styleProp) {
20490             return Roo.fly(el).getStyle(styleProp);
20491         },
20492
20493         /**
20494          * Gets the scrollTop
20495          * @method getScrollTop
20496          * @return {int} the document's scrollTop
20497          * @static
20498          */
20499         getScrollTop: function () { return this.getScroll().top; },
20500
20501         /**
20502          * Gets the scrollLeft
20503          * @method getScrollLeft
20504          * @return {int} the document's scrollTop
20505          * @static
20506          */
20507         getScrollLeft: function () { return this.getScroll().left; },
20508
20509         /**
20510          * Sets the x/y position of an element to the location of the
20511          * target element.
20512          * @method moveToEl
20513          * @param {HTMLElement} moveEl      The element to move
20514          * @param {HTMLElement} targetEl    The position reference element
20515          * @static
20516          */
20517         moveToEl: function (moveEl, targetEl) {
20518             var aCoord = Roo.lib.Dom.getXY(targetEl);
20519             Roo.lib.Dom.setXY(moveEl, aCoord);
20520         },
20521
20522         /**
20523          * Numeric array sort function
20524          * @method numericSort
20525          * @static
20526          */
20527         numericSort: function(a, b) { return (a - b); },
20528
20529         /**
20530          * Internal counter
20531          * @property _timeoutCount
20532          * @private
20533          * @static
20534          */
20535         _timeoutCount: 0,
20536
20537         /**
20538          * Trying to make the load order less important.  Without this we get
20539          * an error if this file is loaded before the Event Utility.
20540          * @method _addListeners
20541          * @private
20542          * @static
20543          */
20544         _addListeners: function() {
20545             var DDM = Roo.dd.DDM;
20546             if ( Roo.lib.Event && document ) {
20547                 DDM._onLoad();
20548             } else {
20549                 if (DDM._timeoutCount > 2000) {
20550                 } else {
20551                     setTimeout(DDM._addListeners, 10);
20552                     if (document && document.body) {
20553                         DDM._timeoutCount += 1;
20554                     }
20555                 }
20556             }
20557         },
20558
20559         /**
20560          * Recursively searches the immediate parent and all child nodes for
20561          * the handle element in order to determine wheter or not it was
20562          * clicked.
20563          * @method handleWasClicked
20564          * @param node the html element to inspect
20565          * @static
20566          */
20567         handleWasClicked: function(node, id) {
20568             if (this.isHandle(id, node.id)) {
20569                 return true;
20570             } else {
20571                 // check to see if this is a text node child of the one we want
20572                 var p = node.parentNode;
20573
20574                 while (p) {
20575                     if (this.isHandle(id, p.id)) {
20576                         return true;
20577                     } else {
20578                         p = p.parentNode;
20579                     }
20580                 }
20581             }
20582
20583             return false;
20584         }
20585
20586     };
20587
20588 }();
20589
20590 // shorter alias, save a few bytes
20591 Roo.dd.DDM = Roo.dd.DragDropMgr;
20592 Roo.dd.DDM._addListeners();
20593
20594 }/*
20595  * Based on:
20596  * Ext JS Library 1.1.1
20597  * Copyright(c) 2006-2007, Ext JS, LLC.
20598  *
20599  * Originally Released Under LGPL - original licence link has changed is not relivant.
20600  *
20601  * Fork - LGPL
20602  * <script type="text/javascript">
20603  */
20604
20605 /**
20606  * @class Roo.dd.DD
20607  * A DragDrop implementation where the linked element follows the
20608  * mouse cursor during a drag.
20609  * @extends Roo.dd.DragDrop
20610  * @constructor
20611  * @param {String} id the id of the linked element
20612  * @param {String} sGroup the group of related DragDrop items
20613  * @param {object} config an object containing configurable attributes
20614  *                Valid properties for DD:
20615  *                    scroll
20616  */
20617 Roo.dd.DD = function(id, sGroup, config) {
20618     if (id) {
20619         this.init(id, sGroup, config);
20620     }
20621 };
20622
20623 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20624
20625     /**
20626      * When set to true, the utility automatically tries to scroll the browser
20627      * window wehn a drag and drop element is dragged near the viewport boundary.
20628      * Defaults to true.
20629      * @property scroll
20630      * @type boolean
20631      */
20632     scroll: true,
20633
20634     /**
20635      * Sets the pointer offset to the distance between the linked element's top
20636      * left corner and the location the element was clicked
20637      * @method autoOffset
20638      * @param {int} iPageX the X coordinate of the click
20639      * @param {int} iPageY the Y coordinate of the click
20640      */
20641     autoOffset: function(iPageX, iPageY) {
20642         var x = iPageX - this.startPageX;
20643         var y = iPageY - this.startPageY;
20644         this.setDelta(x, y);
20645     },
20646
20647     /**
20648      * Sets the pointer offset.  You can call this directly to force the
20649      * offset to be in a particular location (e.g., pass in 0,0 to set it
20650      * to the center of the object)
20651      * @method setDelta
20652      * @param {int} iDeltaX the distance from the left
20653      * @param {int} iDeltaY the distance from the top
20654      */
20655     setDelta: function(iDeltaX, iDeltaY) {
20656         this.deltaX = iDeltaX;
20657         this.deltaY = iDeltaY;
20658     },
20659
20660     /**
20661      * Sets the drag element to the location of the mousedown or click event,
20662      * maintaining the cursor location relative to the location on the element
20663      * that was clicked.  Override this if you want to place the element in a
20664      * location other than where the cursor is.
20665      * @method setDragElPos
20666      * @param {int} iPageX the X coordinate of the mousedown or drag event
20667      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20668      */
20669     setDragElPos: function(iPageX, iPageY) {
20670         // the first time we do this, we are going to check to make sure
20671         // the element has css positioning
20672
20673         var el = this.getDragEl();
20674         this.alignElWithMouse(el, iPageX, iPageY);
20675     },
20676
20677     /**
20678      * Sets the element to the location of the mousedown or click event,
20679      * maintaining the cursor location relative to the location on the element
20680      * that was clicked.  Override this if you want to place the element in a
20681      * location other than where the cursor is.
20682      * @method alignElWithMouse
20683      * @param {HTMLElement} el the element to move
20684      * @param {int} iPageX the X coordinate of the mousedown or drag event
20685      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20686      */
20687     alignElWithMouse: function(el, iPageX, iPageY) {
20688         var oCoord = this.getTargetCoord(iPageX, iPageY);
20689         var fly = el.dom ? el : Roo.fly(el);
20690         if (!this.deltaSetXY) {
20691             var aCoord = [oCoord.x, oCoord.y];
20692             fly.setXY(aCoord);
20693             var newLeft = fly.getLeft(true);
20694             var newTop  = fly.getTop(true);
20695             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20696         } else {
20697             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20698         }
20699
20700         this.cachePosition(oCoord.x, oCoord.y);
20701         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20702         return oCoord;
20703     },
20704
20705     /**
20706      * Saves the most recent position so that we can reset the constraints and
20707      * tick marks on-demand.  We need to know this so that we can calculate the
20708      * number of pixels the element is offset from its original position.
20709      * @method cachePosition
20710      * @param iPageX the current x position (optional, this just makes it so we
20711      * don't have to look it up again)
20712      * @param iPageY the current y position (optional, this just makes it so we
20713      * don't have to look it up again)
20714      */
20715     cachePosition: function(iPageX, iPageY) {
20716         if (iPageX) {
20717             this.lastPageX = iPageX;
20718             this.lastPageY = iPageY;
20719         } else {
20720             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20721             this.lastPageX = aCoord[0];
20722             this.lastPageY = aCoord[1];
20723         }
20724     },
20725
20726     /**
20727      * Auto-scroll the window if the dragged object has been moved beyond the
20728      * visible window boundary.
20729      * @method autoScroll
20730      * @param {int} x the drag element's x position
20731      * @param {int} y the drag element's y position
20732      * @param {int} h the height of the drag element
20733      * @param {int} w the width of the drag element
20734      * @private
20735      */
20736     autoScroll: function(x, y, h, w) {
20737
20738         if (this.scroll) {
20739             // The client height
20740             var clientH = Roo.lib.Dom.getViewWidth();
20741
20742             // The client width
20743             var clientW = Roo.lib.Dom.getViewHeight();
20744
20745             // The amt scrolled down
20746             var st = this.DDM.getScrollTop();
20747
20748             // The amt scrolled right
20749             var sl = this.DDM.getScrollLeft();
20750
20751             // Location of the bottom of the element
20752             var bot = h + y;
20753
20754             // Location of the right of the element
20755             var right = w + x;
20756
20757             // The distance from the cursor to the bottom of the visible area,
20758             // adjusted so that we don't scroll if the cursor is beyond the
20759             // element drag constraints
20760             var toBot = (clientH + st - y - this.deltaY);
20761
20762             // The distance from the cursor to the right of the visible area
20763             var toRight = (clientW + sl - x - this.deltaX);
20764
20765
20766             // How close to the edge the cursor must be before we scroll
20767             // var thresh = (document.all) ? 100 : 40;
20768             var thresh = 40;
20769
20770             // How many pixels to scroll per autoscroll op.  This helps to reduce
20771             // clunky scrolling. IE is more sensitive about this ... it needs this
20772             // value to be higher.
20773             var scrAmt = (document.all) ? 80 : 30;
20774
20775             // Scroll down if we are near the bottom of the visible page and the
20776             // obj extends below the crease
20777             if ( bot > clientH && toBot < thresh ) {
20778                 window.scrollTo(sl, st + scrAmt);
20779             }
20780
20781             // Scroll up if the window is scrolled down and the top of the object
20782             // goes above the top border
20783             if ( y < st && st > 0 && y - st < thresh ) {
20784                 window.scrollTo(sl, st - scrAmt);
20785             }
20786
20787             // Scroll right if the obj is beyond the right border and the cursor is
20788             // near the border.
20789             if ( right > clientW && toRight < thresh ) {
20790                 window.scrollTo(sl + scrAmt, st);
20791             }
20792
20793             // Scroll left if the window has been scrolled to the right and the obj
20794             // extends past the left border
20795             if ( x < sl && sl > 0 && x - sl < thresh ) {
20796                 window.scrollTo(sl - scrAmt, st);
20797             }
20798         }
20799     },
20800
20801     /**
20802      * Finds the location the element should be placed if we want to move
20803      * it to where the mouse location less the click offset would place us.
20804      * @method getTargetCoord
20805      * @param {int} iPageX the X coordinate of the click
20806      * @param {int} iPageY the Y coordinate of the click
20807      * @return an object that contains the coordinates (Object.x and Object.y)
20808      * @private
20809      */
20810     getTargetCoord: function(iPageX, iPageY) {
20811
20812
20813         var x = iPageX - this.deltaX;
20814         var y = iPageY - this.deltaY;
20815
20816         if (this.constrainX) {
20817             if (x < this.minX) { x = this.minX; }
20818             if (x > this.maxX) { x = this.maxX; }
20819         }
20820
20821         if (this.constrainY) {
20822             if (y < this.minY) { y = this.minY; }
20823             if (y > this.maxY) { y = this.maxY; }
20824         }
20825
20826         x = this.getTick(x, this.xTicks);
20827         y = this.getTick(y, this.yTicks);
20828
20829
20830         return {x:x, y:y};
20831     },
20832
20833     /*
20834      * Sets up config options specific to this class. Overrides
20835      * Roo.dd.DragDrop, but all versions of this method through the
20836      * inheritance chain are called
20837      */
20838     applyConfig: function() {
20839         Roo.dd.DD.superclass.applyConfig.call(this);
20840         this.scroll = (this.config.scroll !== false);
20841     },
20842
20843     /*
20844      * Event that fires prior to the onMouseDown event.  Overrides
20845      * Roo.dd.DragDrop.
20846      */
20847     b4MouseDown: function(e) {
20848         // this.resetConstraints();
20849         this.autoOffset(e.getPageX(),
20850                             e.getPageY());
20851     },
20852
20853     /*
20854      * Event that fires prior to the onDrag event.  Overrides
20855      * Roo.dd.DragDrop.
20856      */
20857     b4Drag: function(e) {
20858         this.setDragElPos(e.getPageX(),
20859                             e.getPageY());
20860     },
20861
20862     toString: function() {
20863         return ("DD " + this.id);
20864     }
20865
20866     //////////////////////////////////////////////////////////////////////////
20867     // Debugging ygDragDrop events that can be overridden
20868     //////////////////////////////////////////////////////////////////////////
20869     /*
20870     startDrag: function(x, y) {
20871     },
20872
20873     onDrag: function(e) {
20874     },
20875
20876     onDragEnter: function(e, id) {
20877     },
20878
20879     onDragOver: function(e, id) {
20880     },
20881
20882     onDragOut: function(e, id) {
20883     },
20884
20885     onDragDrop: function(e, id) {
20886     },
20887
20888     endDrag: function(e) {
20889     }
20890
20891     */
20892
20893 });/*
20894  * Based on:
20895  * Ext JS Library 1.1.1
20896  * Copyright(c) 2006-2007, Ext JS, LLC.
20897  *
20898  * Originally Released Under LGPL - original licence link has changed is not relivant.
20899  *
20900  * Fork - LGPL
20901  * <script type="text/javascript">
20902  */
20903
20904 /**
20905  * @class Roo.dd.DDProxy
20906  * A DragDrop implementation that inserts an empty, bordered div into
20907  * the document that follows the cursor during drag operations.  At the time of
20908  * the click, the frame div is resized to the dimensions of the linked html
20909  * element, and moved to the exact location of the linked element.
20910  *
20911  * References to the "frame" element refer to the single proxy element that
20912  * was created to be dragged in place of all DDProxy elements on the
20913  * page.
20914  *
20915  * @extends Roo.dd.DD
20916  * @constructor
20917  * @param {String} id the id of the linked html element
20918  * @param {String} sGroup the group of related DragDrop objects
20919  * @param {object} config an object containing configurable attributes
20920  *                Valid properties for DDProxy in addition to those in DragDrop:
20921  *                   resizeFrame, centerFrame, dragElId
20922  */
20923 Roo.dd.DDProxy = function(id, sGroup, config) {
20924     if (id) {
20925         this.init(id, sGroup, config);
20926         this.initFrame();
20927     }
20928 };
20929
20930 /**
20931  * The default drag frame div id
20932  * @property Roo.dd.DDProxy.dragElId
20933  * @type String
20934  * @static
20935  */
20936 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20937
20938 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20939
20940     /**
20941      * By default we resize the drag frame to be the same size as the element
20942      * we want to drag (this is to get the frame effect).  We can turn it off
20943      * if we want a different behavior.
20944      * @property resizeFrame
20945      * @type boolean
20946      */
20947     resizeFrame: true,
20948
20949     /**
20950      * By default the frame is positioned exactly where the drag element is, so
20951      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20952      * you do not have constraints on the obj is to have the drag frame centered
20953      * around the cursor.  Set centerFrame to true for this effect.
20954      * @property centerFrame
20955      * @type boolean
20956      */
20957     centerFrame: false,
20958
20959     /**
20960      * Creates the proxy element if it does not yet exist
20961      * @method createFrame
20962      */
20963     createFrame: function() {
20964         var self = this;
20965         var body = document.body;
20966
20967         if (!body || !body.firstChild) {
20968             setTimeout( function() { self.createFrame(); }, 50 );
20969             return;
20970         }
20971
20972         var div = this.getDragEl();
20973
20974         if (!div) {
20975             div    = document.createElement("div");
20976             div.id = this.dragElId;
20977             var s  = div.style;
20978
20979             s.position   = "absolute";
20980             s.visibility = "hidden";
20981             s.cursor     = "move";
20982             s.border     = "2px solid #aaa";
20983             s.zIndex     = 999;
20984
20985             // appendChild can blow up IE if invoked prior to the window load event
20986             // while rendering a table.  It is possible there are other scenarios
20987             // that would cause this to happen as well.
20988             body.insertBefore(div, body.firstChild);
20989         }
20990     },
20991
20992     /**
20993      * Initialization for the drag frame element.  Must be called in the
20994      * constructor of all subclasses
20995      * @method initFrame
20996      */
20997     initFrame: function() {
20998         this.createFrame();
20999     },
21000
21001     applyConfig: function() {
21002         Roo.dd.DDProxy.superclass.applyConfig.call(this);
21003
21004         this.resizeFrame = (this.config.resizeFrame !== false);
21005         this.centerFrame = (this.config.centerFrame);
21006         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21007     },
21008
21009     /**
21010      * Resizes the drag frame to the dimensions of the clicked object, positions
21011      * it over the object, and finally displays it
21012      * @method showFrame
21013      * @param {int} iPageX X click position
21014      * @param {int} iPageY Y click position
21015      * @private
21016      */
21017     showFrame: function(iPageX, iPageY) {
21018         var el = this.getEl();
21019         var dragEl = this.getDragEl();
21020         var s = dragEl.style;
21021
21022         this._resizeProxy();
21023
21024         if (this.centerFrame) {
21025             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21026                            Math.round(parseInt(s.height, 10)/2) );
21027         }
21028
21029         this.setDragElPos(iPageX, iPageY);
21030
21031         Roo.fly(dragEl).show();
21032     },
21033
21034     /**
21035      * The proxy is automatically resized to the dimensions of the linked
21036      * element when a drag is initiated, unless resizeFrame is set to false
21037      * @method _resizeProxy
21038      * @private
21039      */
21040     _resizeProxy: function() {
21041         if (this.resizeFrame) {
21042             var el = this.getEl();
21043             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21044         }
21045     },
21046
21047     // overrides Roo.dd.DragDrop
21048     b4MouseDown: function(e) {
21049         var x = e.getPageX();
21050         var y = e.getPageY();
21051         this.autoOffset(x, y);
21052         this.setDragElPos(x, y);
21053     },
21054
21055     // overrides Roo.dd.DragDrop
21056     b4StartDrag: function(x, y) {
21057         // show the drag frame
21058         this.showFrame(x, y);
21059     },
21060
21061     // overrides Roo.dd.DragDrop
21062     b4EndDrag: function(e) {
21063         Roo.fly(this.getDragEl()).hide();
21064     },
21065
21066     // overrides Roo.dd.DragDrop
21067     // By default we try to move the element to the last location of the frame.
21068     // This is so that the default behavior mirrors that of Roo.dd.DD.
21069     endDrag: function(e) {
21070
21071         var lel = this.getEl();
21072         var del = this.getDragEl();
21073
21074         // Show the drag frame briefly so we can get its position
21075         del.style.visibility = "";
21076
21077         this.beforeMove();
21078         // Hide the linked element before the move to get around a Safari
21079         // rendering bug.
21080         lel.style.visibility = "hidden";
21081         Roo.dd.DDM.moveToEl(lel, del);
21082         del.style.visibility = "hidden";
21083         lel.style.visibility = "";
21084
21085         this.afterDrag();
21086     },
21087
21088     beforeMove : function(){
21089
21090     },
21091
21092     afterDrag : function(){
21093
21094     },
21095
21096     toString: function() {
21097         return ("DDProxy " + this.id);
21098     }
21099
21100 });
21101 /*
21102  * Based on:
21103  * Ext JS Library 1.1.1
21104  * Copyright(c) 2006-2007, Ext JS, LLC.
21105  *
21106  * Originally Released Under LGPL - original licence link has changed is not relivant.
21107  *
21108  * Fork - LGPL
21109  * <script type="text/javascript">
21110  */
21111
21112  /**
21113  * @class Roo.dd.DDTarget
21114  * A DragDrop implementation that does not move, but can be a drop
21115  * target.  You would get the same result by simply omitting implementation
21116  * for the event callbacks, but this way we reduce the processing cost of the
21117  * event listener and the callbacks.
21118  * @extends Roo.dd.DragDrop
21119  * @constructor
21120  * @param {String} id the id of the element that is a drop target
21121  * @param {String} sGroup the group of related DragDrop objects
21122  * @param {object} config an object containing configurable attributes
21123  *                 Valid properties for DDTarget in addition to those in
21124  *                 DragDrop:
21125  *                    none
21126  */
21127 Roo.dd.DDTarget = function(id, sGroup, config) {
21128     if (id) {
21129         this.initTarget(id, sGroup, config);
21130     }
21131     if (config && (config.listeners || config.events)) { 
21132         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21133             listeners : config.listeners || {}, 
21134             events : config.events || {} 
21135         });    
21136     }
21137 };
21138
21139 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21140 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21141     toString: function() {
21142         return ("DDTarget " + this.id);
21143     }
21144 });
21145 /*
21146  * Based on:
21147  * Ext JS Library 1.1.1
21148  * Copyright(c) 2006-2007, Ext JS, LLC.
21149  *
21150  * Originally Released Under LGPL - original licence link has changed is not relivant.
21151  *
21152  * Fork - LGPL
21153  * <script type="text/javascript">
21154  */
21155  
21156
21157 /**
21158  * @class Roo.dd.ScrollManager
21159  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21160  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21161  * @singleton
21162  */
21163 Roo.dd.ScrollManager = function(){
21164     var ddm = Roo.dd.DragDropMgr;
21165     var els = {};
21166     var dragEl = null;
21167     var proc = {};
21168     
21169     
21170     
21171     var onStop = function(e){
21172         dragEl = null;
21173         clearProc();
21174     };
21175     
21176     var triggerRefresh = function(){
21177         if(ddm.dragCurrent){
21178              ddm.refreshCache(ddm.dragCurrent.groups);
21179         }
21180     };
21181     
21182     var doScroll = function(){
21183         if(ddm.dragCurrent){
21184             var dds = Roo.dd.ScrollManager;
21185             if(!dds.animate){
21186                 if(proc.el.scroll(proc.dir, dds.increment)){
21187                     triggerRefresh();
21188                 }
21189             }else{
21190                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21191             }
21192         }
21193     };
21194     
21195     var clearProc = function(){
21196         if(proc.id){
21197             clearInterval(proc.id);
21198         }
21199         proc.id = 0;
21200         proc.el = null;
21201         proc.dir = "";
21202     };
21203     
21204     var startProc = function(el, dir){
21205          Roo.log('scroll startproc');
21206         clearProc();
21207         proc.el = el;
21208         proc.dir = dir;
21209         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21210     };
21211     
21212     var onFire = function(e, isDrop){
21213        
21214         if(isDrop || !ddm.dragCurrent){ return; }
21215         var dds = Roo.dd.ScrollManager;
21216         if(!dragEl || dragEl != ddm.dragCurrent){
21217             dragEl = ddm.dragCurrent;
21218             // refresh regions on drag start
21219             dds.refreshCache();
21220         }
21221         
21222         var xy = Roo.lib.Event.getXY(e);
21223         var pt = new Roo.lib.Point(xy[0], xy[1]);
21224         for(var id in els){
21225             var el = els[id], r = el._region;
21226             if(r && r.contains(pt) && el.isScrollable()){
21227                 if(r.bottom - pt.y <= dds.thresh){
21228                     if(proc.el != el){
21229                         startProc(el, "down");
21230                     }
21231                     return;
21232                 }else if(r.right - pt.x <= dds.thresh){
21233                     if(proc.el != el){
21234                         startProc(el, "left");
21235                     }
21236                     return;
21237                 }else if(pt.y - r.top <= dds.thresh){
21238                     if(proc.el != el){
21239                         startProc(el, "up");
21240                     }
21241                     return;
21242                 }else if(pt.x - r.left <= dds.thresh){
21243                     if(proc.el != el){
21244                         startProc(el, "right");
21245                     }
21246                     return;
21247                 }
21248             }
21249         }
21250         clearProc();
21251     };
21252     
21253     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21254     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21255     
21256     return {
21257         /**
21258          * Registers new overflow element(s) to auto scroll
21259          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21260          */
21261         register : function(el){
21262             if(el instanceof Array){
21263                 for(var i = 0, len = el.length; i < len; i++) {
21264                         this.register(el[i]);
21265                 }
21266             }else{
21267                 el = Roo.get(el);
21268                 els[el.id] = el;
21269             }
21270             Roo.dd.ScrollManager.els = els;
21271         },
21272         
21273         /**
21274          * Unregisters overflow element(s) so they are no longer scrolled
21275          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21276          */
21277         unregister : function(el){
21278             if(el instanceof Array){
21279                 for(var i = 0, len = el.length; i < len; i++) {
21280                         this.unregister(el[i]);
21281                 }
21282             }else{
21283                 el = Roo.get(el);
21284                 delete els[el.id];
21285             }
21286         },
21287         
21288         /**
21289          * The number of pixels from the edge of a container the pointer needs to be to 
21290          * trigger scrolling (defaults to 25)
21291          * @type Number
21292          */
21293         thresh : 25,
21294         
21295         /**
21296          * The number of pixels to scroll in each scroll increment (defaults to 50)
21297          * @type Number
21298          */
21299         increment : 100,
21300         
21301         /**
21302          * The frequency of scrolls in milliseconds (defaults to 500)
21303          * @type Number
21304          */
21305         frequency : 500,
21306         
21307         /**
21308          * True to animate the scroll (defaults to true)
21309          * @type Boolean
21310          */
21311         animate: true,
21312         
21313         /**
21314          * The animation duration in seconds - 
21315          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21316          * @type Number
21317          */
21318         animDuration: .4,
21319         
21320         /**
21321          * Manually trigger a cache refresh.
21322          */
21323         refreshCache : function(){
21324             for(var id in els){
21325                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21326                     els[id]._region = els[id].getRegion();
21327                 }
21328             }
21329         }
21330     };
21331 }();/*
21332  * Based on:
21333  * Ext JS Library 1.1.1
21334  * Copyright(c) 2006-2007, Ext JS, LLC.
21335  *
21336  * Originally Released Under LGPL - original licence link has changed is not relivant.
21337  *
21338  * Fork - LGPL
21339  * <script type="text/javascript">
21340  */
21341  
21342
21343 /**
21344  * @class Roo.dd.Registry
21345  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21346  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21347  * @singleton
21348  */
21349 Roo.dd.Registry = function(){
21350     var elements = {}; 
21351     var handles = {}; 
21352     var autoIdSeed = 0;
21353
21354     var getId = function(el, autogen){
21355         if(typeof el == "string"){
21356             return el;
21357         }
21358         var id = el.id;
21359         if(!id && autogen !== false){
21360             id = "roodd-" + (++autoIdSeed);
21361             el.id = id;
21362         }
21363         return id;
21364     };
21365     
21366     return {
21367     /**
21368      * Register a drag drop element
21369      * @param {String|HTMLElement} element The id or DOM node to register
21370      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21371      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21372      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21373      * populated in the data object (if applicable):
21374      * <pre>
21375 Value      Description<br />
21376 ---------  ------------------------------------------<br />
21377 handles    Array of DOM nodes that trigger dragging<br />
21378            for the element being registered<br />
21379 isHandle   True if the element passed in triggers<br />
21380            dragging itself, else false
21381 </pre>
21382      */
21383         register : function(el, data){
21384             data = data || {};
21385             if(typeof el == "string"){
21386                 el = document.getElementById(el);
21387             }
21388             data.ddel = el;
21389             elements[getId(el)] = data;
21390             if(data.isHandle !== false){
21391                 handles[data.ddel.id] = data;
21392             }
21393             if(data.handles){
21394                 var hs = data.handles;
21395                 for(var i = 0, len = hs.length; i < len; i++){
21396                         handles[getId(hs[i])] = data;
21397                 }
21398             }
21399         },
21400
21401     /**
21402      * Unregister a drag drop element
21403      * @param {String|HTMLElement}  element The id or DOM node to unregister
21404      */
21405         unregister : function(el){
21406             var id = getId(el, false);
21407             var data = elements[id];
21408             if(data){
21409                 delete elements[id];
21410                 if(data.handles){
21411                     var hs = data.handles;
21412                     for(var i = 0, len = hs.length; i < len; i++){
21413                         delete handles[getId(hs[i], false)];
21414                     }
21415                 }
21416             }
21417         },
21418
21419     /**
21420      * Returns the handle registered for a DOM Node by id
21421      * @param {String|HTMLElement} id The DOM node or id to look up
21422      * @return {Object} handle The custom handle data
21423      */
21424         getHandle : function(id){
21425             if(typeof id != "string"){ // must be element?
21426                 id = id.id;
21427             }
21428             return handles[id];
21429         },
21430
21431     /**
21432      * Returns the handle that is registered for the DOM node that is the target of the event
21433      * @param {Event} e The event
21434      * @return {Object} handle The custom handle data
21435      */
21436         getHandleFromEvent : function(e){
21437             var t = Roo.lib.Event.getTarget(e);
21438             return t ? handles[t.id] : null;
21439         },
21440
21441     /**
21442      * Returns a custom data object that is registered for a DOM node by id
21443      * @param {String|HTMLElement} id The DOM node or id to look up
21444      * @return {Object} data The custom data
21445      */
21446         getTarget : function(id){
21447             if(typeof id != "string"){ // must be element?
21448                 id = id.id;
21449             }
21450             return elements[id];
21451         },
21452
21453     /**
21454      * Returns a custom data object that is registered for the DOM node that is the target of the event
21455      * @param {Event} e The event
21456      * @return {Object} data The custom data
21457      */
21458         getTargetFromEvent : function(e){
21459             var t = Roo.lib.Event.getTarget(e);
21460             return t ? elements[t.id] || handles[t.id] : null;
21461         }
21462     };
21463 }();/*
21464  * Based on:
21465  * Ext JS Library 1.1.1
21466  * Copyright(c) 2006-2007, Ext JS, LLC.
21467  *
21468  * Originally Released Under LGPL - original licence link has changed is not relivant.
21469  *
21470  * Fork - LGPL
21471  * <script type="text/javascript">
21472  */
21473  
21474
21475 /**
21476  * @class Roo.dd.StatusProxy
21477  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21478  * default drag proxy used by all Roo.dd components.
21479  * @constructor
21480  * @param {Object} config
21481  */
21482 Roo.dd.StatusProxy = function(config){
21483     Roo.apply(this, config);
21484     this.id = this.id || Roo.id();
21485     this.el = new Roo.Layer({
21486         dh: {
21487             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21488                 {tag: "div", cls: "x-dd-drop-icon"},
21489                 {tag: "div", cls: "x-dd-drag-ghost"}
21490             ]
21491         }, 
21492         shadow: !config || config.shadow !== false
21493     });
21494     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21495     this.dropStatus = this.dropNotAllowed;
21496 };
21497
21498 Roo.dd.StatusProxy.prototype = {
21499     /**
21500      * @cfg {String} dropAllowed
21501      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21502      */
21503     dropAllowed : "x-dd-drop-ok",
21504     /**
21505      * @cfg {String} dropNotAllowed
21506      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21507      */
21508     dropNotAllowed : "x-dd-drop-nodrop",
21509
21510     /**
21511      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21512      * over the current target element.
21513      * @param {String} cssClass The css class for the new drop status indicator image
21514      */
21515     setStatus : function(cssClass){
21516         cssClass = cssClass || this.dropNotAllowed;
21517         if(this.dropStatus != cssClass){
21518             this.el.replaceClass(this.dropStatus, cssClass);
21519             this.dropStatus = cssClass;
21520         }
21521     },
21522
21523     /**
21524      * Resets the status indicator to the default dropNotAllowed value
21525      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21526      */
21527     reset : function(clearGhost){
21528         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21529         this.dropStatus = this.dropNotAllowed;
21530         if(clearGhost){
21531             this.ghost.update("");
21532         }
21533     },
21534
21535     /**
21536      * Updates the contents of the ghost element
21537      * @param {String} html The html that will replace the current innerHTML of the ghost element
21538      */
21539     update : function(html){
21540         if(typeof html == "string"){
21541             this.ghost.update(html);
21542         }else{
21543             this.ghost.update("");
21544             html.style.margin = "0";
21545             this.ghost.dom.appendChild(html);
21546         }
21547         // ensure float = none set?? cant remember why though.
21548         var el = this.ghost.dom.firstChild;
21549                 if(el){
21550                         Roo.fly(el).setStyle('float', 'none');
21551                 }
21552     },
21553     
21554     /**
21555      * Returns the underlying proxy {@link Roo.Layer}
21556      * @return {Roo.Layer} el
21557     */
21558     getEl : function(){
21559         return this.el;
21560     },
21561
21562     /**
21563      * Returns the ghost element
21564      * @return {Roo.Element} el
21565      */
21566     getGhost : function(){
21567         return this.ghost;
21568     },
21569
21570     /**
21571      * Hides the proxy
21572      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21573      */
21574     hide : function(clear){
21575         this.el.hide();
21576         if(clear){
21577             this.reset(true);
21578         }
21579     },
21580
21581     /**
21582      * Stops the repair animation if it's currently running
21583      */
21584     stop : function(){
21585         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21586             this.anim.stop();
21587         }
21588     },
21589
21590     /**
21591      * Displays this proxy
21592      */
21593     show : function(){
21594         this.el.show();
21595     },
21596
21597     /**
21598      * Force the Layer to sync its shadow and shim positions to the element
21599      */
21600     sync : function(){
21601         this.el.sync();
21602     },
21603
21604     /**
21605      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21606      * invalid drop operation by the item being dragged.
21607      * @param {Array} xy The XY position of the element ([x, y])
21608      * @param {Function} callback The function to call after the repair is complete
21609      * @param {Object} scope The scope in which to execute the callback
21610      */
21611     repair : function(xy, callback, scope){
21612         this.callback = callback;
21613         this.scope = scope;
21614         if(xy && this.animRepair !== false){
21615             this.el.addClass("x-dd-drag-repair");
21616             this.el.hideUnders(true);
21617             this.anim = this.el.shift({
21618                 duration: this.repairDuration || .5,
21619                 easing: 'easeOut',
21620                 xy: xy,
21621                 stopFx: true,
21622                 callback: this.afterRepair,
21623                 scope: this
21624             });
21625         }else{
21626             this.afterRepair();
21627         }
21628     },
21629
21630     // private
21631     afterRepair : function(){
21632         this.hide(true);
21633         if(typeof this.callback == "function"){
21634             this.callback.call(this.scope || this);
21635         }
21636         this.callback = null;
21637         this.scope = null;
21638     }
21639 };/*
21640  * Based on:
21641  * Ext JS Library 1.1.1
21642  * Copyright(c) 2006-2007, Ext JS, LLC.
21643  *
21644  * Originally Released Under LGPL - original licence link has changed is not relivant.
21645  *
21646  * Fork - LGPL
21647  * <script type="text/javascript">
21648  */
21649
21650 /**
21651  * @class Roo.dd.DragSource
21652  * @extends Roo.dd.DDProxy
21653  * A simple class that provides the basic implementation needed to make any element draggable.
21654  * @constructor
21655  * @param {String/HTMLElement/Element} el The container element
21656  * @param {Object} config
21657  */
21658 Roo.dd.DragSource = function(el, config){
21659     this.el = Roo.get(el);
21660     this.dragData = {};
21661     
21662     Roo.apply(this, config);
21663     
21664     if(!this.proxy){
21665         this.proxy = new Roo.dd.StatusProxy();
21666     }
21667
21668     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21669           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21670     
21671     this.dragging = false;
21672 };
21673
21674 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21675     /**
21676      * @cfg {String} dropAllowed
21677      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21678      */
21679     dropAllowed : "x-dd-drop-ok",
21680     /**
21681      * @cfg {String} dropNotAllowed
21682      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21683      */
21684     dropNotAllowed : "x-dd-drop-nodrop",
21685
21686     /**
21687      * Returns the data object associated with this drag source
21688      * @return {Object} data An object containing arbitrary data
21689      */
21690     getDragData : function(e){
21691         return this.dragData;
21692     },
21693
21694     // private
21695     onDragEnter : function(e, id){
21696         var target = Roo.dd.DragDropMgr.getDDById(id);
21697         this.cachedTarget = target;
21698         if(this.beforeDragEnter(target, e, id) !== false){
21699             if(target.isNotifyTarget){
21700                 var status = target.notifyEnter(this, e, this.dragData);
21701                 this.proxy.setStatus(status);
21702             }else{
21703                 this.proxy.setStatus(this.dropAllowed);
21704             }
21705             
21706             if(this.afterDragEnter){
21707                 /**
21708                  * An empty function by default, but provided so that you can perform a custom action
21709                  * when the dragged item enters the drop target by providing an implementation.
21710                  * @param {Roo.dd.DragDrop} target The drop target
21711                  * @param {Event} e The event object
21712                  * @param {String} id The id of the dragged element
21713                  * @method afterDragEnter
21714                  */
21715                 this.afterDragEnter(target, e, id);
21716             }
21717         }
21718     },
21719
21720     /**
21721      * An empty function by default, but provided so that you can perform a custom action
21722      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21723      * @param {Roo.dd.DragDrop} target The drop target
21724      * @param {Event} e The event object
21725      * @param {String} id The id of the dragged element
21726      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21727      */
21728     beforeDragEnter : function(target, e, id){
21729         return true;
21730     },
21731
21732     // private
21733     alignElWithMouse: function() {
21734         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21735         this.proxy.sync();
21736     },
21737
21738     // private
21739     onDragOver : function(e, id){
21740         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21741         if(this.beforeDragOver(target, e, id) !== false){
21742             if(target.isNotifyTarget){
21743                 var status = target.notifyOver(this, e, this.dragData);
21744                 this.proxy.setStatus(status);
21745             }
21746
21747             if(this.afterDragOver){
21748                 /**
21749                  * An empty function by default, but provided so that you can perform a custom action
21750                  * while the dragged item is over the drop target by providing an implementation.
21751                  * @param {Roo.dd.DragDrop} target The drop target
21752                  * @param {Event} e The event object
21753                  * @param {String} id The id of the dragged element
21754                  * @method afterDragOver
21755                  */
21756                 this.afterDragOver(target, e, id);
21757             }
21758         }
21759     },
21760
21761     /**
21762      * An empty function by default, but provided so that you can perform a custom action
21763      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21764      * @param {Roo.dd.DragDrop} target The drop target
21765      * @param {Event} e The event object
21766      * @param {String} id The id of the dragged element
21767      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21768      */
21769     beforeDragOver : function(target, e, id){
21770         return true;
21771     },
21772
21773     // private
21774     onDragOut : function(e, id){
21775         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21776         if(this.beforeDragOut(target, e, id) !== false){
21777             if(target.isNotifyTarget){
21778                 target.notifyOut(this, e, this.dragData);
21779             }
21780             this.proxy.reset();
21781             if(this.afterDragOut){
21782                 /**
21783                  * An empty function by default, but provided so that you can perform a custom action
21784                  * after the dragged item is dragged out of the target without dropping.
21785                  * @param {Roo.dd.DragDrop} target The drop target
21786                  * @param {Event} e The event object
21787                  * @param {String} id The id of the dragged element
21788                  * @method afterDragOut
21789                  */
21790                 this.afterDragOut(target, e, id);
21791             }
21792         }
21793         this.cachedTarget = null;
21794     },
21795
21796     /**
21797      * An empty function by default, but provided so that you can perform a custom action before the dragged
21798      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21799      * @param {Roo.dd.DragDrop} target The drop target
21800      * @param {Event} e The event object
21801      * @param {String} id The id of the dragged element
21802      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21803      */
21804     beforeDragOut : function(target, e, id){
21805         return true;
21806     },
21807     
21808     // private
21809     onDragDrop : function(e, id){
21810         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21811         if(this.beforeDragDrop(target, e, id) !== false){
21812             if(target.isNotifyTarget){
21813                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21814                     this.onValidDrop(target, e, id);
21815                 }else{
21816                     this.onInvalidDrop(target, e, id);
21817                 }
21818             }else{
21819                 this.onValidDrop(target, e, id);
21820             }
21821             
21822             if(this.afterDragDrop){
21823                 /**
21824                  * An empty function by default, but provided so that you can perform a custom action
21825                  * after a valid drag drop has occurred by providing an implementation.
21826                  * @param {Roo.dd.DragDrop} target The drop target
21827                  * @param {Event} e The event object
21828                  * @param {String} id The id of the dropped element
21829                  * @method afterDragDrop
21830                  */
21831                 this.afterDragDrop(target, e, id);
21832             }
21833         }
21834         delete this.cachedTarget;
21835     },
21836
21837     /**
21838      * An empty function by default, but provided so that you can perform a custom action before the dragged
21839      * item is dropped onto the target and optionally cancel the onDragDrop.
21840      * @param {Roo.dd.DragDrop} target The drop target
21841      * @param {Event} e The event object
21842      * @param {String} id The id of the dragged element
21843      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21844      */
21845     beforeDragDrop : function(target, e, id){
21846         return true;
21847     },
21848
21849     // private
21850     onValidDrop : function(target, e, id){
21851         this.hideProxy();
21852         if(this.afterValidDrop){
21853             /**
21854              * An empty function by default, but provided so that you can perform a custom action
21855              * after a valid drop has occurred by providing an implementation.
21856              * @param {Object} target The target DD 
21857              * @param {Event} e The event object
21858              * @param {String} id The id of the dropped element
21859              * @method afterInvalidDrop
21860              */
21861             this.afterValidDrop(target, e, id);
21862         }
21863     },
21864
21865     // private
21866     getRepairXY : function(e, data){
21867         return this.el.getXY();  
21868     },
21869
21870     // private
21871     onInvalidDrop : function(target, e, id){
21872         this.beforeInvalidDrop(target, e, id);
21873         if(this.cachedTarget){
21874             if(this.cachedTarget.isNotifyTarget){
21875                 this.cachedTarget.notifyOut(this, e, this.dragData);
21876             }
21877             this.cacheTarget = null;
21878         }
21879         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21880
21881         if(this.afterInvalidDrop){
21882             /**
21883              * An empty function by default, but provided so that you can perform a custom action
21884              * after an invalid drop has occurred by providing an implementation.
21885              * @param {Event} e The event object
21886              * @param {String} id The id of the dropped element
21887              * @method afterInvalidDrop
21888              */
21889             this.afterInvalidDrop(e, id);
21890         }
21891     },
21892
21893     // private
21894     afterRepair : function(){
21895         if(Roo.enableFx){
21896             this.el.highlight(this.hlColor || "c3daf9");
21897         }
21898         this.dragging = false;
21899     },
21900
21901     /**
21902      * An empty function by default, but provided so that you can perform a custom action after an invalid
21903      * drop has occurred.
21904      * @param {Roo.dd.DragDrop} target The drop target
21905      * @param {Event} e The event object
21906      * @param {String} id The id of the dragged element
21907      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21908      */
21909     beforeInvalidDrop : function(target, e, id){
21910         return true;
21911     },
21912
21913     // private
21914     handleMouseDown : function(e){
21915         if(this.dragging) {
21916             return;
21917         }
21918         var data = this.getDragData(e);
21919         if(data && this.onBeforeDrag(data, e) !== false){
21920             this.dragData = data;
21921             this.proxy.stop();
21922             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21923         } 
21924     },
21925
21926     /**
21927      * An empty function by default, but provided so that you can perform a custom action before the initial
21928      * drag event begins and optionally cancel it.
21929      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21930      * @param {Event} e The event object
21931      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21932      */
21933     onBeforeDrag : function(data, e){
21934         return true;
21935     },
21936
21937     /**
21938      * An empty function by default, but provided so that you can perform a custom action once the initial
21939      * drag event has begun.  The drag cannot be canceled from this function.
21940      * @param {Number} x The x position of the click on the dragged object
21941      * @param {Number} y The y position of the click on the dragged object
21942      */
21943     onStartDrag : Roo.emptyFn,
21944
21945     // private - YUI override
21946     startDrag : function(x, y){
21947         this.proxy.reset();
21948         this.dragging = true;
21949         this.proxy.update("");
21950         this.onInitDrag(x, y);
21951         this.proxy.show();
21952     },
21953
21954     // private
21955     onInitDrag : function(x, y){
21956         var clone = this.el.dom.cloneNode(true);
21957         clone.id = Roo.id(); // prevent duplicate ids
21958         this.proxy.update(clone);
21959         this.onStartDrag(x, y);
21960         return true;
21961     },
21962
21963     /**
21964      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21965      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21966      */
21967     getProxy : function(){
21968         return this.proxy;  
21969     },
21970
21971     /**
21972      * Hides the drag source's {@link Roo.dd.StatusProxy}
21973      */
21974     hideProxy : function(){
21975         this.proxy.hide();  
21976         this.proxy.reset(true);
21977         this.dragging = false;
21978     },
21979
21980     // private
21981     triggerCacheRefresh : function(){
21982         Roo.dd.DDM.refreshCache(this.groups);
21983     },
21984
21985     // private - override to prevent hiding
21986     b4EndDrag: function(e) {
21987     },
21988
21989     // private - override to prevent moving
21990     endDrag : function(e){
21991         this.onEndDrag(this.dragData, e);
21992     },
21993
21994     // private
21995     onEndDrag : function(data, e){
21996     },
21997     
21998     // private - pin to cursor
21999     autoOffset : function(x, y) {
22000         this.setDelta(-12, -20);
22001     }    
22002 });/*
22003  * Based on:
22004  * Ext JS Library 1.1.1
22005  * Copyright(c) 2006-2007, Ext JS, LLC.
22006  *
22007  * Originally Released Under LGPL - original licence link has changed is not relivant.
22008  *
22009  * Fork - LGPL
22010  * <script type="text/javascript">
22011  */
22012
22013
22014 /**
22015  * @class Roo.dd.DropTarget
22016  * @extends Roo.dd.DDTarget
22017  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22018  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22019  * @constructor
22020  * @param {String/HTMLElement/Element} el The container element
22021  * @param {Object} config
22022  */
22023 Roo.dd.DropTarget = function(el, config){
22024     this.el = Roo.get(el);
22025     
22026     var listeners = false; ;
22027     if (config && config.listeners) {
22028         listeners= config.listeners;
22029         delete config.listeners;
22030     }
22031     Roo.apply(this, config);
22032     
22033     if(this.containerScroll){
22034         Roo.dd.ScrollManager.register(this.el);
22035     }
22036     this.addEvents( {
22037          /**
22038          * @scope Roo.dd.DropTarget
22039          */
22040          
22041          /**
22042          * @event enter
22043          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22044          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22045          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22046          * 
22047          * IMPORTANT : it should set this.overClass and this.dropAllowed
22048          * 
22049          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22050          * @param {Event} e The event
22051          * @param {Object} data An object containing arbitrary data supplied by the drag source
22052          */
22053         "enter" : true,
22054         
22055          /**
22056          * @event over
22057          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22058          * This method will be called on every mouse movement while the drag source is over the drop target.
22059          * This default implementation simply returns the dropAllowed config value.
22060          * 
22061          * IMPORTANT : it should set this.dropAllowed
22062          * 
22063          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22064          * @param {Event} e The event
22065          * @param {Object} data An object containing arbitrary data supplied by the drag source
22066          
22067          */
22068         "over" : true,
22069         /**
22070          * @event out
22071          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22072          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22073          * overClass (if any) from the drop element.
22074          * 
22075          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22076          * @param {Event} e The event
22077          * @param {Object} data An object containing arbitrary data supplied by the drag source
22078          */
22079          "out" : true,
22080          
22081         /**
22082          * @event drop
22083          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22084          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22085          * implementation that does something to process the drop event and returns true so that the drag source's
22086          * repair action does not run.
22087          * 
22088          * IMPORTANT : it should set this.success
22089          * 
22090          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22091          * @param {Event} e The event
22092          * @param {Object} data An object containing arbitrary data supplied by the drag source
22093         */
22094          "drop" : true
22095     });
22096             
22097      
22098     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22099         this.el.dom, 
22100         this.ddGroup || this.group,
22101         {
22102             isTarget: true,
22103             listeners : listeners || {} 
22104            
22105         
22106         }
22107     );
22108
22109 };
22110
22111 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22112     /**
22113      * @cfg {String} overClass
22114      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22115      */
22116      /**
22117      * @cfg {String} ddGroup
22118      * The drag drop group to handle drop events for
22119      */
22120      
22121     /**
22122      * @cfg {String} dropAllowed
22123      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22124      */
22125     dropAllowed : "x-dd-drop-ok",
22126     /**
22127      * @cfg {String} dropNotAllowed
22128      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22129      */
22130     dropNotAllowed : "x-dd-drop-nodrop",
22131     /**
22132      * @cfg {boolean} success
22133      * set this after drop listener.. 
22134      */
22135     success : false,
22136     /**
22137      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22138      * if the drop point is valid for over/enter..
22139      */
22140     valid : false,
22141     // private
22142     isTarget : true,
22143
22144     // private
22145     isNotifyTarget : true,
22146     
22147     /**
22148      * @hide
22149      */
22150     notifyEnter : function(dd, e, data)
22151     {
22152         this.valid = true;
22153         this.fireEvent('enter', dd, e, data);
22154         if(this.overClass){
22155             this.el.addClass(this.overClass);
22156         }
22157         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22158             this.valid ? this.dropAllowed : this.dropNotAllowed
22159         );
22160     },
22161
22162     /**
22163      * @hide
22164      */
22165     notifyOver : function(dd, e, data)
22166     {
22167         this.valid = true;
22168         this.fireEvent('over', dd, e, data);
22169         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22170             this.valid ? this.dropAllowed : this.dropNotAllowed
22171         );
22172     },
22173
22174     /**
22175      * @hide
22176      */
22177     notifyOut : function(dd, e, data)
22178     {
22179         this.fireEvent('out', dd, e, data);
22180         if(this.overClass){
22181             this.el.removeClass(this.overClass);
22182         }
22183     },
22184
22185     /**
22186      * @hide
22187      */
22188     notifyDrop : function(dd, e, data)
22189     {
22190         this.success = false;
22191         this.fireEvent('drop', dd, e, data);
22192         return this.success;
22193     }
22194 });/*
22195  * Based on:
22196  * Ext JS Library 1.1.1
22197  * Copyright(c) 2006-2007, Ext JS, LLC.
22198  *
22199  * Originally Released Under LGPL - original licence link has changed is not relivant.
22200  *
22201  * Fork - LGPL
22202  * <script type="text/javascript">
22203  */
22204
22205
22206 /**
22207  * @class Roo.dd.DragZone
22208  * @extends Roo.dd.DragSource
22209  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22210  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22211  * @constructor
22212  * @param {String/HTMLElement/Element} el The container element
22213  * @param {Object} config
22214  */
22215 Roo.dd.DragZone = function(el, config){
22216     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22217     if(this.containerScroll){
22218         Roo.dd.ScrollManager.register(this.el);
22219     }
22220 };
22221
22222 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22223     /**
22224      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22225      * for auto scrolling during drag operations.
22226      */
22227     /**
22228      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22229      * method after a failed drop (defaults to "c3daf9" - light blue)
22230      */
22231
22232     /**
22233      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22234      * for a valid target to drag based on the mouse down. Override this method
22235      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22236      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22237      * @param {EventObject} e The mouse down event
22238      * @return {Object} The dragData
22239      */
22240     getDragData : function(e){
22241         return Roo.dd.Registry.getHandleFromEvent(e);
22242     },
22243     
22244     /**
22245      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22246      * this.dragData.ddel
22247      * @param {Number} x The x position of the click on the dragged object
22248      * @param {Number} y The y position of the click on the dragged object
22249      * @return {Boolean} true to continue the drag, false to cancel
22250      */
22251     onInitDrag : function(x, y){
22252         this.proxy.update(this.dragData.ddel.cloneNode(true));
22253         this.onStartDrag(x, y);
22254         return true;
22255     },
22256     
22257     /**
22258      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22259      */
22260     afterRepair : function(){
22261         if(Roo.enableFx){
22262             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22263         }
22264         this.dragging = false;
22265     },
22266
22267     /**
22268      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22269      * the XY of this.dragData.ddel
22270      * @param {EventObject} e The mouse up event
22271      * @return {Array} The xy location (e.g. [100, 200])
22272      */
22273     getRepairXY : function(e){
22274         return Roo.Element.fly(this.dragData.ddel).getXY();  
22275     }
22276 });/*
22277  * Based on:
22278  * Ext JS Library 1.1.1
22279  * Copyright(c) 2006-2007, Ext JS, LLC.
22280  *
22281  * Originally Released Under LGPL - original licence link has changed is not relivant.
22282  *
22283  * Fork - LGPL
22284  * <script type="text/javascript">
22285  */
22286 /**
22287  * @class Roo.dd.DropZone
22288  * @extends Roo.dd.DropTarget
22289  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22290  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22291  * @constructor
22292  * @param {String/HTMLElement/Element} el The container element
22293  * @param {Object} config
22294  */
22295 Roo.dd.DropZone = function(el, config){
22296     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22297 };
22298
22299 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22300     /**
22301      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22302      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22303      * provide your own custom lookup.
22304      * @param {Event} e The event
22305      * @return {Object} data The custom data
22306      */
22307     getTargetFromEvent : function(e){
22308         return Roo.dd.Registry.getTargetFromEvent(e);
22309     },
22310
22311     /**
22312      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22313      * that it has registered.  This method has no default implementation and should be overridden to provide
22314      * node-specific processing if necessary.
22315      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22316      * {@link #getTargetFromEvent} for this node)
22317      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22318      * @param {Event} e The event
22319      * @param {Object} data An object containing arbitrary data supplied by the drag source
22320      */
22321     onNodeEnter : function(n, dd, e, data){
22322         
22323     },
22324
22325     /**
22326      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22327      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22328      * overridden to provide the proper feedback.
22329      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22330      * {@link #getTargetFromEvent} for this node)
22331      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22332      * @param {Event} e The event
22333      * @param {Object} data An object containing arbitrary data supplied by the drag source
22334      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22335      * underlying {@link Roo.dd.StatusProxy} can be updated
22336      */
22337     onNodeOver : function(n, dd, e, data){
22338         return this.dropAllowed;
22339     },
22340
22341     /**
22342      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22343      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22344      * node-specific processing if necessary.
22345      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22346      * {@link #getTargetFromEvent} for this node)
22347      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22348      * @param {Event} e The event
22349      * @param {Object} data An object containing arbitrary data supplied by the drag source
22350      */
22351     onNodeOut : function(n, dd, e, data){
22352         
22353     },
22354
22355     /**
22356      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22357      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22358      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22359      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22360      * {@link #getTargetFromEvent} for this node)
22361      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22362      * @param {Event} e The event
22363      * @param {Object} data An object containing arbitrary data supplied by the drag source
22364      * @return {Boolean} True if the drop was valid, else false
22365      */
22366     onNodeDrop : function(n, dd, e, data){
22367         return false;
22368     },
22369
22370     /**
22371      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22372      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22373      * it should be overridden to provide the proper feedback if necessary.
22374      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22375      * @param {Event} e The event
22376      * @param {Object} data An object containing arbitrary data supplied by the drag source
22377      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22378      * underlying {@link Roo.dd.StatusProxy} can be updated
22379      */
22380     onContainerOver : function(dd, e, data){
22381         return this.dropNotAllowed;
22382     },
22383
22384     /**
22385      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22386      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22387      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22388      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22389      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22390      * @param {Event} e The event
22391      * @param {Object} data An object containing arbitrary data supplied by the drag source
22392      * @return {Boolean} True if the drop was valid, else false
22393      */
22394     onContainerDrop : function(dd, e, data){
22395         return false;
22396     },
22397
22398     /**
22399      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22400      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22401      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22402      * you should override this method and provide a custom implementation.
22403      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22404      * @param {Event} e The event
22405      * @param {Object} data An object containing arbitrary data supplied by the drag source
22406      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22407      * underlying {@link Roo.dd.StatusProxy} can be updated
22408      */
22409     notifyEnter : function(dd, e, data){
22410         return this.dropNotAllowed;
22411     },
22412
22413     /**
22414      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22415      * This method will be called on every mouse movement while the drag source is over the drop zone.
22416      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22417      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22418      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22419      * registered node, it will call {@link #onContainerOver}.
22420      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22421      * @param {Event} e The event
22422      * @param {Object} data An object containing arbitrary data supplied by the drag source
22423      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22424      * underlying {@link Roo.dd.StatusProxy} can be updated
22425      */
22426     notifyOver : function(dd, e, data){
22427         var n = this.getTargetFromEvent(e);
22428         if(!n){ // not over valid drop target
22429             if(this.lastOverNode){
22430                 this.onNodeOut(this.lastOverNode, dd, e, data);
22431                 this.lastOverNode = null;
22432             }
22433             return this.onContainerOver(dd, e, data);
22434         }
22435         if(this.lastOverNode != n){
22436             if(this.lastOverNode){
22437                 this.onNodeOut(this.lastOverNode, dd, e, data);
22438             }
22439             this.onNodeEnter(n, dd, e, data);
22440             this.lastOverNode = n;
22441         }
22442         return this.onNodeOver(n, dd, e, data);
22443     },
22444
22445     /**
22446      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22447      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22448      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22449      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22450      * @param {Event} e The event
22451      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22452      */
22453     notifyOut : function(dd, e, data){
22454         if(this.lastOverNode){
22455             this.onNodeOut(this.lastOverNode, dd, e, data);
22456             this.lastOverNode = null;
22457         }
22458     },
22459
22460     /**
22461      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22462      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22463      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22464      * otherwise it will call {@link #onContainerDrop}.
22465      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22466      * @param {Event} e The event
22467      * @param {Object} data An object containing arbitrary data supplied by the drag source
22468      * @return {Boolean} True if the drop was valid, else false
22469      */
22470     notifyDrop : function(dd, e, data){
22471         if(this.lastOverNode){
22472             this.onNodeOut(this.lastOverNode, dd, e, data);
22473             this.lastOverNode = null;
22474         }
22475         var n = this.getTargetFromEvent(e);
22476         return n ?
22477             this.onNodeDrop(n, dd, e, data) :
22478             this.onContainerDrop(dd, e, data);
22479     },
22480
22481     // private
22482     triggerCacheRefresh : function(){
22483         Roo.dd.DDM.refreshCache(this.groups);
22484     }  
22485 });/*
22486  * Based on:
22487  * Ext JS Library 1.1.1
22488  * Copyright(c) 2006-2007, Ext JS, LLC.
22489  *
22490  * Originally Released Under LGPL - original licence link has changed is not relivant.
22491  *
22492  * Fork - LGPL
22493  * <script type="text/javascript">
22494  */
22495
22496
22497 /**
22498  * @class Roo.data.SortTypes
22499  * @singleton
22500  * Defines the default sorting (casting?) comparison functions used when sorting data.
22501  */
22502 Roo.data.SortTypes = {
22503     /**
22504      * Default sort that does nothing
22505      * @param {Mixed} s The value being converted
22506      * @return {Mixed} The comparison value
22507      */
22508     none : function(s){
22509         return s;
22510     },
22511     
22512     /**
22513      * The regular expression used to strip tags
22514      * @type {RegExp}
22515      * @property
22516      */
22517     stripTagsRE : /<\/?[^>]+>/gi,
22518     
22519     /**
22520      * Strips all HTML tags to sort on text only
22521      * @param {Mixed} s The value being converted
22522      * @return {String} The comparison value
22523      */
22524     asText : function(s){
22525         return String(s).replace(this.stripTagsRE, "");
22526     },
22527     
22528     /**
22529      * Strips all HTML tags to sort on text only - Case insensitive
22530      * @param {Mixed} s The value being converted
22531      * @return {String} The comparison value
22532      */
22533     asUCText : function(s){
22534         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22535     },
22536     
22537     /**
22538      * Case insensitive string
22539      * @param {Mixed} s The value being converted
22540      * @return {String} The comparison value
22541      */
22542     asUCString : function(s) {
22543         return String(s).toUpperCase();
22544     },
22545     
22546     /**
22547      * Date sorting
22548      * @param {Mixed} s The value being converted
22549      * @return {Number} The comparison value
22550      */
22551     asDate : function(s) {
22552         if(!s){
22553             return 0;
22554         }
22555         if(s instanceof Date){
22556             return s.getTime();
22557         }
22558         return Date.parse(String(s));
22559     },
22560     
22561     /**
22562      * Float sorting
22563      * @param {Mixed} s The value being converted
22564      * @return {Float} The comparison value
22565      */
22566     asFloat : function(s) {
22567         var val = parseFloat(String(s).replace(/,/g, ""));
22568         if(isNaN(val)) {
22569             val = 0;
22570         }
22571         return val;
22572     },
22573     
22574     /**
22575      * Integer sorting
22576      * @param {Mixed} s The value being converted
22577      * @return {Number} The comparison value
22578      */
22579     asInt : function(s) {
22580         var val = parseInt(String(s).replace(/,/g, ""));
22581         if(isNaN(val)) {
22582             val = 0;
22583         }
22584         return val;
22585     }
22586 };/*
22587  * Based on:
22588  * Ext JS Library 1.1.1
22589  * Copyright(c) 2006-2007, Ext JS, LLC.
22590  *
22591  * Originally Released Under LGPL - original licence link has changed is not relivant.
22592  *
22593  * Fork - LGPL
22594  * <script type="text/javascript">
22595  */
22596
22597 /**
22598 * @class Roo.data.Record
22599  * Instances of this class encapsulate both record <em>definition</em> information, and record
22600  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22601  * to access Records cached in an {@link Roo.data.Store} object.<br>
22602  * <p>
22603  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22604  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22605  * objects.<br>
22606  * <p>
22607  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22608  * @constructor
22609  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22610  * {@link #create}. The parameters are the same.
22611  * @param {Array} data An associative Array of data values keyed by the field name.
22612  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22613  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22614  * not specified an integer id is generated.
22615  */
22616 Roo.data.Record = function(data, id){
22617     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22618     this.data = data;
22619 };
22620
22621 /**
22622  * Generate a constructor for a specific record layout.
22623  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22624  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22625  * Each field definition object may contain the following properties: <ul>
22626  * <li><b>name</b> : String<p style="margin-left:1em">The name by which the field is referenced within the Record. This is referenced by,
22627  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22628  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22629  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22630  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22631  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22632  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22633  * this may be omitted.</p></li>
22634  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22635  * <ul><li>auto (Default, implies no conversion)</li>
22636  * <li>string</li>
22637  * <li>int</li>
22638  * <li>float</li>
22639  * <li>boolean</li>
22640  * <li>date</li></ul></p></li>
22641  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22642  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22643  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22644  * by the Reader into an object that will be stored in the Record. It is passed the
22645  * following parameters:<ul>
22646  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22647  * </ul></p></li>
22648  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22649  * </ul>
22650  * <br>usage:<br><pre><code>
22651 var TopicRecord = Roo.data.Record.create(
22652     {name: 'title', mapping: 'topic_title'},
22653     {name: 'author', mapping: 'username'},
22654     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22655     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22656     {name: 'lastPoster', mapping: 'user2'},
22657     {name: 'excerpt', mapping: 'post_text'}
22658 );
22659
22660 var myNewRecord = new TopicRecord({
22661     title: 'Do my job please',
22662     author: 'noobie',
22663     totalPosts: 1,
22664     lastPost: new Date(),
22665     lastPoster: 'Animal',
22666     excerpt: 'No way dude!'
22667 });
22668 myStore.add(myNewRecord);
22669 </code></pre>
22670  * @method create
22671  * @static
22672  */
22673 Roo.data.Record.create = function(o){
22674     var f = function(){
22675         f.superclass.constructor.apply(this, arguments);
22676     };
22677     Roo.extend(f, Roo.data.Record);
22678     var p = f.prototype;
22679     p.fields = new Roo.util.MixedCollection(false, function(field){
22680         return field.name;
22681     });
22682     for(var i = 0, len = o.length; i < len; i++){
22683         p.fields.add(new Roo.data.Field(o[i]));
22684     }
22685     f.getField = function(name){
22686         return p.fields.get(name);  
22687     };
22688     return f;
22689 };
22690
22691 Roo.data.Record.AUTO_ID = 1000;
22692 Roo.data.Record.EDIT = 'edit';
22693 Roo.data.Record.REJECT = 'reject';
22694 Roo.data.Record.COMMIT = 'commit';
22695
22696 Roo.data.Record.prototype = {
22697     /**
22698      * Readonly flag - true if this record has been modified.
22699      * @type Boolean
22700      */
22701     dirty : false,
22702     editing : false,
22703     error: null,
22704     modified: null,
22705
22706     // private
22707     join : function(store){
22708         this.store = store;
22709     },
22710
22711     /**
22712      * Set the named field to the specified value.
22713      * @param {String} name The name of the field to set.
22714      * @param {Object} value The value to set the field to.
22715      */
22716     set : function(name, value){
22717         if(this.data[name] == value){
22718             return;
22719         }
22720         this.dirty = true;
22721         if(!this.modified){
22722             this.modified = {};
22723         }
22724         if(typeof this.modified[name] == 'undefined'){
22725             this.modified[name] = this.data[name];
22726         }
22727         this.data[name] = value;
22728         if(!this.editing && this.store){
22729             this.store.afterEdit(this);
22730         }       
22731     },
22732
22733     /**
22734      * Get the value of the named field.
22735      * @param {String} name The name of the field to get the value of.
22736      * @return {Object} The value of the field.
22737      */
22738     get : function(name){
22739         return this.data[name]; 
22740     },
22741
22742     // private
22743     beginEdit : function(){
22744         this.editing = true;
22745         this.modified = {}; 
22746     },
22747
22748     // private
22749     cancelEdit : function(){
22750         this.editing = false;
22751         delete this.modified;
22752     },
22753
22754     // private
22755     endEdit : function(){
22756         this.editing = false;
22757         if(this.dirty && this.store){
22758             this.store.afterEdit(this);
22759         }
22760     },
22761
22762     /**
22763      * Usually called by the {@link Roo.data.Store} which owns the Record.
22764      * Rejects all changes made to the Record since either creation, or the last commit operation.
22765      * Modified fields are reverted to their original values.
22766      * <p>
22767      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22768      * of reject operations.
22769      */
22770     reject : function(){
22771         var m = this.modified;
22772         for(var n in m){
22773             if(typeof m[n] != "function"){
22774                 this.data[n] = m[n];
22775             }
22776         }
22777         this.dirty = false;
22778         delete this.modified;
22779         this.editing = false;
22780         if(this.store){
22781             this.store.afterReject(this);
22782         }
22783     },
22784
22785     /**
22786      * Usually called by the {@link Roo.data.Store} which owns the Record.
22787      * Commits all changes made to the Record since either creation, or the last commit operation.
22788      * <p>
22789      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22790      * of commit operations.
22791      */
22792     commit : function(){
22793         this.dirty = false;
22794         delete this.modified;
22795         this.editing = false;
22796         if(this.store){
22797             this.store.afterCommit(this);
22798         }
22799     },
22800
22801     // private
22802     hasError : function(){
22803         return this.error != null;
22804     },
22805
22806     // private
22807     clearError : function(){
22808         this.error = null;
22809     },
22810
22811     /**
22812      * Creates a copy of this record.
22813      * @param {String} id (optional) A new record id if you don't want to use this record's id
22814      * @return {Record}
22815      */
22816     copy : function(newId) {
22817         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22818     }
22819 };/*
22820  * Based on:
22821  * Ext JS Library 1.1.1
22822  * Copyright(c) 2006-2007, Ext JS, LLC.
22823  *
22824  * Originally Released Under LGPL - original licence link has changed is not relivant.
22825  *
22826  * Fork - LGPL
22827  * <script type="text/javascript">
22828  */
22829
22830
22831
22832 /**
22833  * @class Roo.data.Store
22834  * @extends Roo.util.Observable
22835  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22836  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22837  * <p>
22838  * A Store object uses an implementation of {@link Roo.data.DataProxy} to access a data object unless you call loadData() directly and pass in your data. The Store object
22839  * has no knowledge of the format of the data returned by the Proxy.<br>
22840  * <p>
22841  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22842  * instances from the data object. These records are cached and made available through accessor functions.
22843  * @constructor
22844  * Creates a new Store.
22845  * @param {Object} config A config object containing the objects needed for the Store to access data,
22846  * and read the data into Records.
22847  */
22848 Roo.data.Store = function(config){
22849     this.data = new Roo.util.MixedCollection(false);
22850     this.data.getKey = function(o){
22851         return o.id;
22852     };
22853     this.baseParams = {};
22854     // private
22855     this.paramNames = {
22856         "start" : "start",
22857         "limit" : "limit",
22858         "sort" : "sort",
22859         "dir" : "dir",
22860         "multisort" : "_multisort"
22861     };
22862
22863     if(config && config.data){
22864         this.inlineData = config.data;
22865         delete config.data;
22866     }
22867
22868     Roo.apply(this, config);
22869     
22870     if(this.reader){ // reader passed
22871         this.reader = Roo.factory(this.reader, Roo.data);
22872         this.reader.xmodule = this.xmodule || false;
22873         if(!this.recordType){
22874             this.recordType = this.reader.recordType;
22875         }
22876         if(this.reader.onMetaChange){
22877             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22878         }
22879     }
22880
22881     if(this.recordType){
22882         this.fields = this.recordType.prototype.fields;
22883     }
22884     this.modified = [];
22885
22886     this.addEvents({
22887         /**
22888          * @event datachanged
22889          * Fires when the data cache has changed, and a widget which is using this Store
22890          * as a Record cache should refresh its view.
22891          * @param {Store} this
22892          */
22893         datachanged : true,
22894         /**
22895          * @event metachange
22896          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22897          * @param {Store} this
22898          * @param {Object} meta The JSON metadata
22899          */
22900         metachange : true,
22901         /**
22902          * @event add
22903          * Fires when Records have been added to the Store
22904          * @param {Store} this
22905          * @param {Roo.data.Record[]} records The array of Records added
22906          * @param {Number} index The index at which the record(s) were added
22907          */
22908         add : true,
22909         /**
22910          * @event remove
22911          * Fires when a Record has been removed from the Store
22912          * @param {Store} this
22913          * @param {Roo.data.Record} record The Record that was removed
22914          * @param {Number} index The index at which the record was removed
22915          */
22916         remove : true,
22917         /**
22918          * @event update
22919          * Fires when a Record has been updated
22920          * @param {Store} this
22921          * @param {Roo.data.Record} record The Record that was updated
22922          * @param {String} operation The update operation being performed.  Value may be one of:
22923          * <pre><code>
22924  Roo.data.Record.EDIT
22925  Roo.data.Record.REJECT
22926  Roo.data.Record.COMMIT
22927          * </code></pre>
22928          */
22929         update : true,
22930         /**
22931          * @event clear
22932          * Fires when the data cache has been cleared.
22933          * @param {Store} this
22934          */
22935         clear : true,
22936         /**
22937          * @event beforeload
22938          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22939          * the load action will be canceled.
22940          * @param {Store} this
22941          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22942          */
22943         beforeload : true,
22944         /**
22945          * @event beforeloadadd
22946          * Fires after a new set of Records has been loaded.
22947          * @param {Store} this
22948          * @param {Roo.data.Record[]} records The Records that were loaded
22949          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22950          */
22951         beforeloadadd : true,
22952         /**
22953          * @event load
22954          * Fires after a new set of Records has been loaded, before they are added to the store.
22955          * @param {Store} this
22956          * @param {Roo.data.Record[]} records The Records that were loaded
22957          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22958          * @params {Object} return from reader
22959          */
22960         load : true,
22961         /**
22962          * @event loadexception
22963          * Fires if an exception occurs in the Proxy during loading.
22964          * Called with the signature of the Proxy's "loadexception" event.
22965          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22966          * 
22967          * @param {Proxy} 
22968          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22969          * @param {Object} load options 
22970          * @param {Object} jsonData from your request (normally this contains the Exception)
22971          */
22972         loadexception : true
22973     });
22974     
22975     if(this.proxy){
22976         this.proxy = Roo.factory(this.proxy, Roo.data);
22977         this.proxy.xmodule = this.xmodule || false;
22978         this.relayEvents(this.proxy,  ["loadexception"]);
22979     }
22980     this.sortToggle = {};
22981     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22982
22983     Roo.data.Store.superclass.constructor.call(this);
22984
22985     if(this.inlineData){
22986         this.loadData(this.inlineData);
22987         delete this.inlineData;
22988     }
22989 };
22990
22991 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22992      /**
22993     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22994     * without a remote query - used by combo/forms at present.
22995     */
22996     
22997     /**
22998     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22999     */
23000     /**
23001     * @cfg {Array} data Inline data to be loaded when the store is initialized.
23002     */
23003     /**
23004     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
23005     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23006     */
23007     /**
23008     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23009     * on any HTTP request
23010     */
23011     /**
23012     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23013     */
23014     /**
23015     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23016     */
23017     multiSort: false,
23018     /**
23019     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23020     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23021     */
23022     remoteSort : false,
23023
23024     /**
23025     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23026      * loaded or when a record is removed. (defaults to false).
23027     */
23028     pruneModifiedRecords : false,
23029
23030     // private
23031     lastOptions : null,
23032
23033     /**
23034      * Add Records to the Store and fires the add event.
23035      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23036      */
23037     add : function(records){
23038         records = [].concat(records);
23039         for(var i = 0, len = records.length; i < len; i++){
23040             records[i].join(this);
23041         }
23042         var index = this.data.length;
23043         this.data.addAll(records);
23044         this.fireEvent("add", this, records, index);
23045     },
23046
23047     /**
23048      * Remove a Record from the Store and fires the remove event.
23049      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23050      */
23051     remove : function(record){
23052         var index = this.data.indexOf(record);
23053         this.data.removeAt(index);
23054  
23055         if(this.pruneModifiedRecords){
23056             this.modified.remove(record);
23057         }
23058         this.fireEvent("remove", this, record, index);
23059     },
23060
23061     /**
23062      * Remove all Records from the Store and fires the clear event.
23063      */
23064     removeAll : function(){
23065         this.data.clear();
23066         if(this.pruneModifiedRecords){
23067             this.modified = [];
23068         }
23069         this.fireEvent("clear", this);
23070     },
23071
23072     /**
23073      * Inserts Records to the Store at the given index and fires the add event.
23074      * @param {Number} index The start index at which to insert the passed Records.
23075      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23076      */
23077     insert : function(index, records){
23078         records = [].concat(records);
23079         for(var i = 0, len = records.length; i < len; i++){
23080             this.data.insert(index, records[i]);
23081             records[i].join(this);
23082         }
23083         this.fireEvent("add", this, records, index);
23084     },
23085
23086     /**
23087      * Get the index within the cache of the passed Record.
23088      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23089      * @return {Number} The index of the passed Record. Returns -1 if not found.
23090      */
23091     indexOf : function(record){
23092         return this.data.indexOf(record);
23093     },
23094
23095     /**
23096      * Get the index within the cache of the Record with the passed id.
23097      * @param {String} id The id of the Record to find.
23098      * @return {Number} The index of the Record. Returns -1 if not found.
23099      */
23100     indexOfId : function(id){
23101         return this.data.indexOfKey(id);
23102     },
23103
23104     /**
23105      * Get the Record with the specified id.
23106      * @param {String} id The id of the Record to find.
23107      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23108      */
23109     getById : function(id){
23110         return this.data.key(id);
23111     },
23112
23113     /**
23114      * Get the Record at the specified index.
23115      * @param {Number} index The index of the Record to find.
23116      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23117      */
23118     getAt : function(index){
23119         return this.data.itemAt(index);
23120     },
23121
23122     /**
23123      * Returns a range of Records between specified indices.
23124      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23125      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23126      * @return {Roo.data.Record[]} An array of Records
23127      */
23128     getRange : function(start, end){
23129         return this.data.getRange(start, end);
23130     },
23131
23132     // private
23133     storeOptions : function(o){
23134         o = Roo.apply({}, o);
23135         delete o.callback;
23136         delete o.scope;
23137         this.lastOptions = o;
23138     },
23139
23140     /**
23141      * Loads the Record cache from the configured Proxy using the configured Reader.
23142      * <p>
23143      * If using remote paging, then the first load call must specify the <em>start</em>
23144      * and <em>limit</em> properties in the options.params property to establish the initial
23145      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23146      * <p>
23147      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23148      * and this call will return before the new data has been loaded. Perform any post-processing
23149      * in a callback function, or in a "load" event handler.</strong>
23150      * <p>
23151      * @param {Object} options An object containing properties which control loading options:<ul>
23152      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23153      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23154      * passed the following arguments:<ul>
23155      * <li>r : Roo.data.Record[]</li>
23156      * <li>options: Options object from the load call</li>
23157      * <li>success: Boolean success indicator</li></ul></li>
23158      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23159      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23160      * </ul>
23161      */
23162     load : function(options){
23163         options = options || {};
23164         if(this.fireEvent("beforeload", this, options) !== false){
23165             this.storeOptions(options);
23166             var p = Roo.apply(options.params || {}, this.baseParams);
23167             // if meta was not loaded from remote source.. try requesting it.
23168             if (!this.reader.metaFromRemote) {
23169                 p._requestMeta = 1;
23170             }
23171             if(this.sortInfo && this.remoteSort){
23172                 var pn = this.paramNames;
23173                 p[pn["sort"]] = this.sortInfo.field;
23174                 p[pn["dir"]] = this.sortInfo.direction;
23175             }
23176             if (this.multiSort) {
23177                 var pn = this.paramNames;
23178                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23179             }
23180             
23181             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23182         }
23183     },
23184
23185     /**
23186      * Reloads the Record cache from the configured Proxy using the configured Reader and
23187      * the options from the last load operation performed.
23188      * @param {Object} options (optional) An object containing properties which may override the options
23189      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23190      * the most recently used options are reused).
23191      */
23192     reload : function(options){
23193         this.load(Roo.applyIf(options||{}, this.lastOptions));
23194     },
23195
23196     // private
23197     // Called as a callback by the Reader during a load operation.
23198     loadRecords : function(o, options, success){
23199         if(!o || success === false){
23200             if(success !== false){
23201                 this.fireEvent("load", this, [], options, o);
23202             }
23203             if(options.callback){
23204                 options.callback.call(options.scope || this, [], options, false);
23205             }
23206             return;
23207         }
23208         // if data returned failure - throw an exception.
23209         if (o.success === false) {
23210             // show a message if no listener is registered.
23211             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23212                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23213             }
23214             // loadmask wil be hooked into this..
23215             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23216             return;
23217         }
23218         var r = o.records, t = o.totalRecords || r.length;
23219         
23220         this.fireEvent("beforeloadadd", this, r, options, o);
23221         
23222         if(!options || options.add !== true){
23223             if(this.pruneModifiedRecords){
23224                 this.modified = [];
23225             }
23226             for(var i = 0, len = r.length; i < len; i++){
23227                 r[i].join(this);
23228             }
23229             if(this.snapshot){
23230                 this.data = this.snapshot;
23231                 delete this.snapshot;
23232             }
23233             this.data.clear();
23234             this.data.addAll(r);
23235             this.totalLength = t;
23236             this.applySort();
23237             this.fireEvent("datachanged", this);
23238         }else{
23239             this.totalLength = Math.max(t, this.data.length+r.length);
23240             this.add(r);
23241         }
23242         
23243         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23244                 
23245             var e = new Roo.data.Record({});
23246
23247             e.set(this.parent.displayField, this.parent.emptyTitle);
23248             e.set(this.parent.valueField, '');
23249
23250             this.insert(0, e);
23251         }
23252             
23253         this.fireEvent("load", this, r, options, o);
23254         if(options.callback){
23255             options.callback.call(options.scope || this, r, options, true);
23256         }
23257     },
23258
23259
23260     /**
23261      * Loads data from a passed data block. A Reader which understands the format of the data
23262      * must have been configured in the constructor.
23263      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23264      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23265      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23266      */
23267     loadData : function(o, append){
23268         var r = this.reader.readRecords(o);
23269         this.loadRecords(r, {add: append}, true);
23270     },
23271
23272     /**
23273      * Gets the number of cached records.
23274      * <p>
23275      * <em>If using paging, this may not be the total size of the dataset. If the data object
23276      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23277      * the data set size</em>
23278      */
23279     getCount : function(){
23280         return this.data.length || 0;
23281     },
23282
23283     /**
23284      * Gets the total number of records in the dataset as returned by the server.
23285      * <p>
23286      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23287      * the dataset size</em>
23288      */
23289     getTotalCount : function(){
23290         return this.totalLength || 0;
23291     },
23292
23293     /**
23294      * Returns the sort state of the Store as an object with two properties:
23295      * <pre><code>
23296  field {String} The name of the field by which the Records are sorted
23297  direction {String} The sort order, "ASC" or "DESC"
23298      * </code></pre>
23299      */
23300     getSortState : function(){
23301         return this.sortInfo;
23302     },
23303
23304     // private
23305     applySort : function(){
23306         if(this.sortInfo && !this.remoteSort){
23307             var s = this.sortInfo, f = s.field;
23308             var st = this.fields.get(f).sortType;
23309             var fn = function(r1, r2){
23310                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23311                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23312             };
23313             this.data.sort(s.direction, fn);
23314             if(this.snapshot && this.snapshot != this.data){
23315                 this.snapshot.sort(s.direction, fn);
23316             }
23317         }
23318     },
23319
23320     /**
23321      * Sets the default sort column and order to be used by the next load operation.
23322      * @param {String} fieldName The name of the field to sort by.
23323      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23324      */
23325     setDefaultSort : function(field, dir){
23326         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23327     },
23328
23329     /**
23330      * Sort the Records.
23331      * If remote sorting is used, the sort is performed on the server, and the cache is
23332      * reloaded. If local sorting is used, the cache is sorted internally.
23333      * @param {String} fieldName The name of the field to sort by.
23334      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23335      */
23336     sort : function(fieldName, dir){
23337         var f = this.fields.get(fieldName);
23338         if(!dir){
23339             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23340             
23341             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23342                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23343             }else{
23344                 dir = f.sortDir;
23345             }
23346         }
23347         this.sortToggle[f.name] = dir;
23348         this.sortInfo = {field: f.name, direction: dir};
23349         if(!this.remoteSort){
23350             this.applySort();
23351             this.fireEvent("datachanged", this);
23352         }else{
23353             this.load(this.lastOptions);
23354         }
23355     },
23356
23357     /**
23358      * Calls the specified function for each of the Records in the cache.
23359      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23360      * Returning <em>false</em> aborts and exits the iteration.
23361      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23362      */
23363     each : function(fn, scope){
23364         this.data.each(fn, scope);
23365     },
23366
23367     /**
23368      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23369      * (e.g., during paging).
23370      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23371      */
23372     getModifiedRecords : function(){
23373         return this.modified;
23374     },
23375
23376     // private
23377     createFilterFn : function(property, value, anyMatch){
23378         if(!value.exec){ // not a regex
23379             value = String(value);
23380             if(value.length == 0){
23381                 return false;
23382             }
23383             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23384         }
23385         return function(r){
23386             return value.test(r.data[property]);
23387         };
23388     },
23389
23390     /**
23391      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23392      * @param {String} property A field on your records
23393      * @param {Number} start The record index to start at (defaults to 0)
23394      * @param {Number} end The last record index to include (defaults to length - 1)
23395      * @return {Number} The sum
23396      */
23397     sum : function(property, start, end){
23398         var rs = this.data.items, v = 0;
23399         start = start || 0;
23400         end = (end || end === 0) ? end : rs.length-1;
23401
23402         for(var i = start; i <= end; i++){
23403             v += (rs[i].data[property] || 0);
23404         }
23405         return v;
23406     },
23407
23408     /**
23409      * Filter the records by a specified property.
23410      * @param {String} field A field on your records
23411      * @param {String/RegExp} value Either a string that the field
23412      * should start with or a RegExp to test against the field
23413      * @param {Boolean} anyMatch True to match any part not just the beginning
23414      */
23415     filter : function(property, value, anyMatch){
23416         var fn = this.createFilterFn(property, value, anyMatch);
23417         return fn ? this.filterBy(fn) : this.clearFilter();
23418     },
23419
23420     /**
23421      * Filter by a function. The specified function will be called with each
23422      * record in this data source. If the function returns true the record is included,
23423      * otherwise it is filtered.
23424      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23425      * @param {Object} scope (optional) The scope of the function (defaults to this)
23426      */
23427     filterBy : function(fn, scope){
23428         this.snapshot = this.snapshot || this.data;
23429         this.data = this.queryBy(fn, scope||this);
23430         this.fireEvent("datachanged", this);
23431     },
23432
23433     /**
23434      * Query the records by a specified property.
23435      * @param {String} field A field on your records
23436      * @param {String/RegExp} value Either a string that the field
23437      * should start with or a RegExp to test against the field
23438      * @param {Boolean} anyMatch True to match any part not just the beginning
23439      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23440      */
23441     query : function(property, value, anyMatch){
23442         var fn = this.createFilterFn(property, value, anyMatch);
23443         return fn ? this.queryBy(fn) : this.data.clone();
23444     },
23445
23446     /**
23447      * Query by a function. The specified function will be called with each
23448      * record in this data source. If the function returns true the record is included
23449      * in the results.
23450      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23451      * @param {Object} scope (optional) The scope of the function (defaults to this)
23452       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23453      **/
23454     queryBy : function(fn, scope){
23455         var data = this.snapshot || this.data;
23456         return data.filterBy(fn, scope||this);
23457     },
23458
23459     /**
23460      * Collects unique values for a particular dataIndex from this store.
23461      * @param {String} dataIndex The property to collect
23462      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23463      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23464      * @return {Array} An array of the unique values
23465      **/
23466     collect : function(dataIndex, allowNull, bypassFilter){
23467         var d = (bypassFilter === true && this.snapshot) ?
23468                 this.snapshot.items : this.data.items;
23469         var v, sv, r = [], l = {};
23470         for(var i = 0, len = d.length; i < len; i++){
23471             v = d[i].data[dataIndex];
23472             sv = String(v);
23473             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23474                 l[sv] = true;
23475                 r[r.length] = v;
23476             }
23477         }
23478         return r;
23479     },
23480
23481     /**
23482      * Revert to a view of the Record cache with no filtering applied.
23483      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23484      */
23485     clearFilter : function(suppressEvent){
23486         if(this.snapshot && this.snapshot != this.data){
23487             this.data = this.snapshot;
23488             delete this.snapshot;
23489             if(suppressEvent !== true){
23490                 this.fireEvent("datachanged", this);
23491             }
23492         }
23493     },
23494
23495     // private
23496     afterEdit : function(record){
23497         if(this.modified.indexOf(record) == -1){
23498             this.modified.push(record);
23499         }
23500         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23501     },
23502     
23503     // private
23504     afterReject : function(record){
23505         this.modified.remove(record);
23506         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23507     },
23508
23509     // private
23510     afterCommit : function(record){
23511         this.modified.remove(record);
23512         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23513     },
23514
23515     /**
23516      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23517      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23518      */
23519     commitChanges : function(){
23520         var m = this.modified.slice(0);
23521         this.modified = [];
23522         for(var i = 0, len = m.length; i < len; i++){
23523             m[i].commit();
23524         }
23525     },
23526
23527     /**
23528      * Cancel outstanding changes on all changed records.
23529      */
23530     rejectChanges : function(){
23531         var m = this.modified.slice(0);
23532         this.modified = [];
23533         for(var i = 0, len = m.length; i < len; i++){
23534             m[i].reject();
23535         }
23536     },
23537
23538     onMetaChange : function(meta, rtype, o){
23539         this.recordType = rtype;
23540         this.fields = rtype.prototype.fields;
23541         delete this.snapshot;
23542         this.sortInfo = meta.sortInfo || this.sortInfo;
23543         this.modified = [];
23544         this.fireEvent('metachange', this, this.reader.meta);
23545     },
23546     
23547     moveIndex : function(data, type)
23548     {
23549         var index = this.indexOf(data);
23550         
23551         var newIndex = index + type;
23552         
23553         this.remove(data);
23554         
23555         this.insert(newIndex, data);
23556         
23557     }
23558 });/*
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  * @class Roo.data.SimpleStore
23571  * @extends Roo.data.Store
23572  * Small helper class to make creating Stores from Array data easier.
23573  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23574  * @cfg {Array} fields An array of field definition objects, or field name strings.
23575  * @cfg {Object} an existing reader (eg. copied from another store)
23576  * @cfg {Array} data The multi-dimensional array of data
23577  * @constructor
23578  * @param {Object} config
23579  */
23580 Roo.data.SimpleStore = function(config)
23581 {
23582     Roo.data.SimpleStore.superclass.constructor.call(this, {
23583         isLocal : true,
23584         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
23585                 id: config.id
23586             },
23587             Roo.data.Record.create(config.fields)
23588         ),
23589         proxy : new Roo.data.MemoryProxy(config.data)
23590     });
23591     this.load();
23592 };
23593 Roo.extend(Roo.data.SimpleStore, 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 /**
23606  * @extends Roo.data.Store
23607  * @class Roo.data.JsonStore
23608  * Small helper class to make creating Stores for JSON data easier. <br/>
23609 <pre><code>
23610 var store = new Roo.data.JsonStore({
23611     url: 'get-images.php',
23612     root: 'images',
23613     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23614 });
23615 </code></pre>
23616  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23617  * JsonReader and HttpProxy (unless inline data is provided).</b>
23618  * @cfg {Array} fields An array of field definition objects, or field name strings.
23619  * @constructor
23620  * @param {Object} config
23621  */
23622 Roo.data.JsonStore = function(c){
23623     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23624         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23625         reader: new Roo.data.JsonReader(c, c.fields)
23626     }));
23627 };
23628 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23629  * Based on:
23630  * Ext JS Library 1.1.1
23631  * Copyright(c) 2006-2007, Ext JS, LLC.
23632  *
23633  * Originally Released Under LGPL - original licence link has changed is not relivant.
23634  *
23635  * Fork - LGPL
23636  * <script type="text/javascript">
23637  */
23638
23639  
23640 Roo.data.Field = function(config){
23641     if(typeof config == "string"){
23642         config = {name: config};
23643     }
23644     Roo.apply(this, config);
23645     
23646     if(!this.type){
23647         this.type = "auto";
23648     }
23649     
23650     var st = Roo.data.SortTypes;
23651     // named sortTypes are supported, here we look them up
23652     if(typeof this.sortType == "string"){
23653         this.sortType = st[this.sortType];
23654     }
23655     
23656     // set default sortType for strings and dates
23657     if(!this.sortType){
23658         switch(this.type){
23659             case "string":
23660                 this.sortType = st.asUCString;
23661                 break;
23662             case "date":
23663                 this.sortType = st.asDate;
23664                 break;
23665             default:
23666                 this.sortType = st.none;
23667         }
23668     }
23669
23670     // define once
23671     var stripRe = /[\$,%]/g;
23672
23673     // prebuilt conversion function for this field, instead of
23674     // switching every time we're reading a value
23675     if(!this.convert){
23676         var cv, dateFormat = this.dateFormat;
23677         switch(this.type){
23678             case "":
23679             case "auto":
23680             case undefined:
23681                 cv = function(v){ return v; };
23682                 break;
23683             case "string":
23684                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23685                 break;
23686             case "int":
23687                 cv = function(v){
23688                     return v !== undefined && v !== null && v !== '' ?
23689                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23690                     };
23691                 break;
23692             case "float":
23693                 cv = function(v){
23694                     return v !== undefined && v !== null && v !== '' ?
23695                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23696                     };
23697                 break;
23698             case "bool":
23699             case "boolean":
23700                 cv = function(v){ return v === true || v === "true" || v == 1; };
23701                 break;
23702             case "date":
23703                 cv = function(v){
23704                     if(!v){
23705                         return '';
23706                     }
23707                     if(v instanceof Date){
23708                         return v;
23709                     }
23710                     if(dateFormat){
23711                         if(dateFormat == "timestamp"){
23712                             return new Date(v*1000);
23713                         }
23714                         return Date.parseDate(v, dateFormat);
23715                     }
23716                     var parsed = Date.parse(v);
23717                     return parsed ? new Date(parsed) : null;
23718                 };
23719              break;
23720             
23721         }
23722         this.convert = cv;
23723     }
23724 };
23725
23726 Roo.data.Field.prototype = {
23727     dateFormat: null,
23728     defaultValue: "",
23729     mapping: null,
23730     sortType : null,
23731     sortDir : "ASC"
23732 };/*
23733  * Based on:
23734  * Ext JS Library 1.1.1
23735  * Copyright(c) 2006-2007, Ext JS, LLC.
23736  *
23737  * Originally Released Under LGPL - original licence link has changed is not relivant.
23738  *
23739  * Fork - LGPL
23740  * <script type="text/javascript">
23741  */
23742  
23743 // Base class for reading structured data from a data source.  This class is intended to be
23744 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23745
23746 /**
23747  * @class Roo.data.DataReader
23748  * Base class for reading structured data from a data source.  This class is intended to be
23749  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23750  */
23751
23752 Roo.data.DataReader = function(meta, recordType){
23753     
23754     this.meta = meta;
23755     
23756     this.recordType = recordType instanceof Array ? 
23757         Roo.data.Record.create(recordType) : recordType;
23758 };
23759
23760 Roo.data.DataReader.prototype = {
23761     
23762     
23763     readerType : 'Data',
23764      /**
23765      * Create an empty record
23766      * @param {Object} data (optional) - overlay some values
23767      * @return {Roo.data.Record} record created.
23768      */
23769     newRow :  function(d) {
23770         var da =  {};
23771         this.recordType.prototype.fields.each(function(c) {
23772             switch( c.type) {
23773                 case 'int' : da[c.name] = 0; break;
23774                 case 'date' : da[c.name] = new Date(); break;
23775                 case 'float' : da[c.name] = 0.0; break;
23776                 case 'boolean' : da[c.name] = false; break;
23777                 default : da[c.name] = ""; break;
23778             }
23779             
23780         });
23781         return new this.recordType(Roo.apply(da, d));
23782     }
23783     
23784     
23785 };/*
23786  * Based on:
23787  * Ext JS Library 1.1.1
23788  * Copyright(c) 2006-2007, Ext JS, LLC.
23789  *
23790  * Originally Released Under LGPL - original licence link has changed is not relivant.
23791  *
23792  * Fork - LGPL
23793  * <script type="text/javascript">
23794  */
23795
23796 /**
23797  * @class Roo.data.DataProxy
23798  * @extends Roo.data.Observable
23799  * This class is an abstract base class for implementations which provide retrieval of
23800  * unformatted data objects.<br>
23801  * <p>
23802  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23803  * (of the appropriate type which knows how to parse the data object) to provide a block of
23804  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23805  * <p>
23806  * Custom implementations must implement the load method as described in
23807  * {@link Roo.data.HttpProxy#load}.
23808  */
23809 Roo.data.DataProxy = function(){
23810     this.addEvents({
23811         /**
23812          * @event beforeload
23813          * Fires before a network request is made to retrieve a data object.
23814          * @param {Object} This DataProxy object.
23815          * @param {Object} params The params parameter to the load function.
23816          */
23817         beforeload : true,
23818         /**
23819          * @event load
23820          * Fires before the load method's callback is called.
23821          * @param {Object} This DataProxy object.
23822          * @param {Object} o The data object.
23823          * @param {Object} arg The callback argument object passed to the load function.
23824          */
23825         load : true,
23826         /**
23827          * @event loadexception
23828          * Fires if an Exception occurs during data retrieval.
23829          * @param {Object} This DataProxy object.
23830          * @param {Object} o The data object.
23831          * @param {Object} arg The callback argument object passed to the load function.
23832          * @param {Object} e The Exception.
23833          */
23834         loadexception : true
23835     });
23836     Roo.data.DataProxy.superclass.constructor.call(this);
23837 };
23838
23839 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23840
23841     /**
23842      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23843      */
23844 /*
23845  * Based on:
23846  * Ext JS Library 1.1.1
23847  * Copyright(c) 2006-2007, Ext JS, LLC.
23848  *
23849  * Originally Released Under LGPL - original licence link has changed is not relivant.
23850  *
23851  * Fork - LGPL
23852  * <script type="text/javascript">
23853  */
23854 /**
23855  * @class Roo.data.MemoryProxy
23856  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23857  * to the Reader when its load method is called.
23858  * @constructor
23859  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23860  */
23861 Roo.data.MemoryProxy = function(data){
23862     if (data.data) {
23863         data = data.data;
23864     }
23865     Roo.data.MemoryProxy.superclass.constructor.call(this);
23866     this.data = data;
23867 };
23868
23869 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23870     
23871     /**
23872      * Load data from the requested source (in this case an in-memory
23873      * data object passed to the constructor), read the data object into
23874      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23875      * process that block using the passed callback.
23876      * @param {Object} params This parameter is not used by the MemoryProxy class.
23877      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23878      * object into a block of Roo.data.Records.
23879      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23880      * The function must be passed <ul>
23881      * <li>The Record block object</li>
23882      * <li>The "arg" argument from the load function</li>
23883      * <li>A boolean success indicator</li>
23884      * </ul>
23885      * @param {Object} scope The scope in which to call the callback
23886      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23887      */
23888     load : function(params, reader, callback, scope, arg){
23889         params = params || {};
23890         var result;
23891         try {
23892             result = reader.readRecords(params.data ? params.data :this.data);
23893         }catch(e){
23894             this.fireEvent("loadexception", this, arg, null, e);
23895             callback.call(scope, null, arg, false);
23896             return;
23897         }
23898         callback.call(scope, result, arg, true);
23899     },
23900     
23901     // private
23902     update : function(params, records){
23903         
23904     }
23905 });/*
23906  * Based on:
23907  * Ext JS Library 1.1.1
23908  * Copyright(c) 2006-2007, Ext JS, LLC.
23909  *
23910  * Originally Released Under LGPL - original licence link has changed is not relivant.
23911  *
23912  * Fork - LGPL
23913  * <script type="text/javascript">
23914  */
23915 /**
23916  * @class Roo.data.HttpProxy
23917  * @extends Roo.data.DataProxy
23918  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23919  * configured to reference a certain URL.<br><br>
23920  * <p>
23921  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23922  * from which the running page was served.<br><br>
23923  * <p>
23924  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23925  * <p>
23926  * Be aware that to enable the browser to parse an XML document, the server must set
23927  * the Content-Type header in the HTTP response to "text/xml".
23928  * @constructor
23929  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23930  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23931  * will be used to make the request.
23932  */
23933 Roo.data.HttpProxy = function(conn){
23934     Roo.data.HttpProxy.superclass.constructor.call(this);
23935     // is conn a conn config or a real conn?
23936     this.conn = conn;
23937     this.useAjax = !conn || !conn.events;
23938   
23939 };
23940
23941 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23942     // thse are take from connection...
23943     
23944     /**
23945      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23946      */
23947     /**
23948      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23949      * extra parameters to each request made by this object. (defaults to undefined)
23950      */
23951     /**
23952      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23953      *  to each request made by this object. (defaults to undefined)
23954      */
23955     /**
23956      * @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)
23957      */
23958     /**
23959      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23960      */
23961      /**
23962      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23963      * @type Boolean
23964      */
23965   
23966
23967     /**
23968      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23969      * @type Boolean
23970      */
23971     /**
23972      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23973      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23974      * a finer-grained basis than the DataProxy events.
23975      */
23976     getConnection : function(){
23977         return this.useAjax ? Roo.Ajax : this.conn;
23978     },
23979
23980     /**
23981      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23982      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23983      * process that block using the passed callback.
23984      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23985      * for the request to the remote server.
23986      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23987      * object into a block of Roo.data.Records.
23988      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23989      * The function must be passed <ul>
23990      * <li>The Record block object</li>
23991      * <li>The "arg" argument from the load function</li>
23992      * <li>A boolean success indicator</li>
23993      * </ul>
23994      * @param {Object} scope The scope in which to call the callback
23995      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23996      */
23997     load : function(params, reader, callback, scope, arg){
23998         if(this.fireEvent("beforeload", this, params) !== false){
23999             var  o = {
24000                 params : params || {},
24001                 request: {
24002                     callback : callback,
24003                     scope : scope,
24004                     arg : arg
24005                 },
24006                 reader: reader,
24007                 callback : this.loadResponse,
24008                 scope: this
24009             };
24010             if(this.useAjax){
24011                 Roo.applyIf(o, this.conn);
24012                 if(this.activeRequest){
24013                     Roo.Ajax.abort(this.activeRequest);
24014                 }
24015                 this.activeRequest = Roo.Ajax.request(o);
24016             }else{
24017                 this.conn.request(o);
24018             }
24019         }else{
24020             callback.call(scope||this, null, arg, false);
24021         }
24022     },
24023
24024     // private
24025     loadResponse : function(o, success, response){
24026         delete this.activeRequest;
24027         if(!success){
24028             this.fireEvent("loadexception", this, o, response);
24029             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24030             return;
24031         }
24032         var result;
24033         try {
24034             result = o.reader.read(response);
24035         }catch(e){
24036             this.fireEvent("loadexception", this, o, response, e);
24037             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24038             return;
24039         }
24040         
24041         this.fireEvent("load", this, o, o.request.arg);
24042         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24043     },
24044
24045     // private
24046     update : function(dataSet){
24047
24048     },
24049
24050     // private
24051     updateResponse : function(dataSet){
24052
24053     }
24054 });/*
24055  * Based on:
24056  * Ext JS Library 1.1.1
24057  * Copyright(c) 2006-2007, Ext JS, LLC.
24058  *
24059  * Originally Released Under LGPL - original licence link has changed is not relivant.
24060  *
24061  * Fork - LGPL
24062  * <script type="text/javascript">
24063  */
24064
24065 /**
24066  * @class Roo.data.ScriptTagProxy
24067  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24068  * other than the originating domain of the running page.<br><br>
24069  * <p>
24070  * <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
24071  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24072  * <p>
24073  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24074  * source code that is used as the source inside a &lt;script> tag.<br><br>
24075  * <p>
24076  * In order for the browser to process the returned data, the server must wrap the data object
24077  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24078  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24079  * depending on whether the callback name was passed:
24080  * <p>
24081  * <pre><code>
24082 boolean scriptTag = false;
24083 String cb = request.getParameter("callback");
24084 if (cb != null) {
24085     scriptTag = true;
24086     response.setContentType("text/javascript");
24087 } else {
24088     response.setContentType("application/x-json");
24089 }
24090 Writer out = response.getWriter();
24091 if (scriptTag) {
24092     out.write(cb + "(");
24093 }
24094 out.print(dataBlock.toJsonString());
24095 if (scriptTag) {
24096     out.write(");");
24097 }
24098 </pre></code>
24099  *
24100  * @constructor
24101  * @param {Object} config A configuration object.
24102  */
24103 Roo.data.ScriptTagProxy = function(config){
24104     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24105     Roo.apply(this, config);
24106     this.head = document.getElementsByTagName("head")[0];
24107 };
24108
24109 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24110
24111 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24112     /**
24113      * @cfg {String} url The URL from which to request the data object.
24114      */
24115     /**
24116      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24117      */
24118     timeout : 30000,
24119     /**
24120      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24121      * the server the name of the callback function set up by the load call to process the returned data object.
24122      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24123      * javascript output which calls this named function passing the data object as its only parameter.
24124      */
24125     callbackParam : "callback",
24126     /**
24127      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24128      * name to the request.
24129      */
24130     nocache : true,
24131
24132     /**
24133      * Load data from the configured URL, read the data object into
24134      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24135      * process that block using the passed callback.
24136      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24137      * for the request to the remote server.
24138      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24139      * object into a block of Roo.data.Records.
24140      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24141      * The function must be passed <ul>
24142      * <li>The Record block object</li>
24143      * <li>The "arg" argument from the load function</li>
24144      * <li>A boolean success indicator</li>
24145      * </ul>
24146      * @param {Object} scope The scope in which to call the callback
24147      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24148      */
24149     load : function(params, reader, callback, scope, arg){
24150         if(this.fireEvent("beforeload", this, params) !== false){
24151
24152             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24153
24154             var url = this.url;
24155             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24156             if(this.nocache){
24157                 url += "&_dc=" + (new Date().getTime());
24158             }
24159             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24160             var trans = {
24161                 id : transId,
24162                 cb : "stcCallback"+transId,
24163                 scriptId : "stcScript"+transId,
24164                 params : params,
24165                 arg : arg,
24166                 url : url,
24167                 callback : callback,
24168                 scope : scope,
24169                 reader : reader
24170             };
24171             var conn = this;
24172
24173             window[trans.cb] = function(o){
24174                 conn.handleResponse(o, trans);
24175             };
24176
24177             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24178
24179             if(this.autoAbort !== false){
24180                 this.abort();
24181             }
24182
24183             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24184
24185             var script = document.createElement("script");
24186             script.setAttribute("src", url);
24187             script.setAttribute("type", "text/javascript");
24188             script.setAttribute("id", trans.scriptId);
24189             this.head.appendChild(script);
24190
24191             this.trans = trans;
24192         }else{
24193             callback.call(scope||this, null, arg, false);
24194         }
24195     },
24196
24197     // private
24198     isLoading : function(){
24199         return this.trans ? true : false;
24200     },
24201
24202     /**
24203      * Abort the current server request.
24204      */
24205     abort : function(){
24206         if(this.isLoading()){
24207             this.destroyTrans(this.trans);
24208         }
24209     },
24210
24211     // private
24212     destroyTrans : function(trans, isLoaded){
24213         this.head.removeChild(document.getElementById(trans.scriptId));
24214         clearTimeout(trans.timeoutId);
24215         if(isLoaded){
24216             window[trans.cb] = undefined;
24217             try{
24218                 delete window[trans.cb];
24219             }catch(e){}
24220         }else{
24221             // if hasn't been loaded, wait for load to remove it to prevent script error
24222             window[trans.cb] = function(){
24223                 window[trans.cb] = undefined;
24224                 try{
24225                     delete window[trans.cb];
24226                 }catch(e){}
24227             };
24228         }
24229     },
24230
24231     // private
24232     handleResponse : function(o, trans){
24233         this.trans = false;
24234         this.destroyTrans(trans, true);
24235         var result;
24236         try {
24237             result = trans.reader.readRecords(o);
24238         }catch(e){
24239             this.fireEvent("loadexception", this, o, trans.arg, e);
24240             trans.callback.call(trans.scope||window, null, trans.arg, false);
24241             return;
24242         }
24243         this.fireEvent("load", this, o, trans.arg);
24244         trans.callback.call(trans.scope||window, result, trans.arg, true);
24245     },
24246
24247     // private
24248     handleFailure : function(trans){
24249         this.trans = false;
24250         this.destroyTrans(trans, false);
24251         this.fireEvent("loadexception", this, null, trans.arg);
24252         trans.callback.call(trans.scope||window, null, trans.arg, false);
24253     }
24254 });/*
24255  * Based on:
24256  * Ext JS Library 1.1.1
24257  * Copyright(c) 2006-2007, Ext JS, LLC.
24258  *
24259  * Originally Released Under LGPL - original licence link has changed is not relivant.
24260  *
24261  * Fork - LGPL
24262  * <script type="text/javascript">
24263  */
24264
24265 /**
24266  * @class Roo.data.JsonReader
24267  * @extends Roo.data.DataReader
24268  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24269  * based on mappings in a provided Roo.data.Record constructor.
24270  * 
24271  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24272  * in the reply previously. 
24273  * 
24274  * <p>
24275  * Example code:
24276  * <pre><code>
24277 var RecordDef = Roo.data.Record.create([
24278     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24279     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24280 ]);
24281 var myReader = new Roo.data.JsonReader({
24282     totalProperty: "results",    // The property which contains the total dataset size (optional)
24283     root: "rows",                // The property which contains an Array of row objects
24284     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24285 }, RecordDef);
24286 </code></pre>
24287  * <p>
24288  * This would consume a JSON file like this:
24289  * <pre><code>
24290 { 'results': 2, 'rows': [
24291     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24292     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24293 }
24294 </code></pre>
24295  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24296  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24297  * paged from the remote server.
24298  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24299  * @cfg {String} root name of the property which contains the Array of row objects.
24300  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24301  * @cfg {Array} fields Array of field definition objects
24302  * @constructor
24303  * Create a new JsonReader
24304  * @param {Object} meta Metadata configuration options
24305  * @param {Object} recordType Either an Array of field definition objects,
24306  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24307  */
24308 Roo.data.JsonReader = function(meta, recordType){
24309     
24310     meta = meta || {};
24311     // set some defaults:
24312     Roo.applyIf(meta, {
24313         totalProperty: 'total',
24314         successProperty : 'success',
24315         root : 'data',
24316         id : 'id'
24317     });
24318     
24319     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24320 };
24321 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24322     
24323     readerType : 'Json',
24324     
24325     /**
24326      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24327      * Used by Store query builder to append _requestMeta to params.
24328      * 
24329      */
24330     metaFromRemote : false,
24331     /**
24332      * This method is only used by a DataProxy which has retrieved data from a remote server.
24333      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24334      * @return {Object} data A data block which is used by an Roo.data.Store object as
24335      * a cache of Roo.data.Records.
24336      */
24337     read : function(response){
24338         var json = response.responseText;
24339        
24340         var o = /* eval:var:o */ eval("("+json+")");
24341         if(!o) {
24342             throw {message: "JsonReader.read: Json object not found"};
24343         }
24344         
24345         if(o.metaData){
24346             
24347             delete this.ef;
24348             this.metaFromRemote = true;
24349             this.meta = o.metaData;
24350             this.recordType = Roo.data.Record.create(o.metaData.fields);
24351             this.onMetaChange(this.meta, this.recordType, o);
24352         }
24353         return this.readRecords(o);
24354     },
24355
24356     // private function a store will implement
24357     onMetaChange : function(meta, recordType, o){
24358
24359     },
24360
24361     /**
24362          * @ignore
24363          */
24364     simpleAccess: function(obj, subsc) {
24365         return obj[subsc];
24366     },
24367
24368         /**
24369          * @ignore
24370          */
24371     getJsonAccessor: function(){
24372         var re = /[\[\.]/;
24373         return function(expr) {
24374             try {
24375                 return(re.test(expr))
24376                     ? new Function("obj", "return obj." + expr)
24377                     : function(obj){
24378                         return obj[expr];
24379                     };
24380             } catch(e){}
24381             return Roo.emptyFn;
24382         };
24383     }(),
24384
24385     /**
24386      * Create a data block containing Roo.data.Records from an XML document.
24387      * @param {Object} o An object which contains an Array of row objects in the property specified
24388      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24389      * which contains the total size of the dataset.
24390      * @return {Object} data A data block which is used by an Roo.data.Store object as
24391      * a cache of Roo.data.Records.
24392      */
24393     readRecords : function(o){
24394         /**
24395          * After any data loads, the raw JSON data is available for further custom processing.
24396          * @type Object
24397          */
24398         this.o = o;
24399         var s = this.meta, Record = this.recordType,
24400             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24401
24402 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24403         if (!this.ef) {
24404             if(s.totalProperty) {
24405                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24406                 }
24407                 if(s.successProperty) {
24408                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24409                 }
24410                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24411                 if (s.id) {
24412                         var g = this.getJsonAccessor(s.id);
24413                         this.getId = function(rec) {
24414                                 var r = g(rec);  
24415                                 return (r === undefined || r === "") ? null : r;
24416                         };
24417                 } else {
24418                         this.getId = function(){return null;};
24419                 }
24420             this.ef = [];
24421             for(var jj = 0; jj < fl; jj++){
24422                 f = fi[jj];
24423                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24424                 this.ef[jj] = this.getJsonAccessor(map);
24425             }
24426         }
24427
24428         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24429         if(s.totalProperty){
24430             var vt = parseInt(this.getTotal(o), 10);
24431             if(!isNaN(vt)){
24432                 totalRecords = vt;
24433             }
24434         }
24435         if(s.successProperty){
24436             var vs = this.getSuccess(o);
24437             if(vs === false || vs === 'false'){
24438                 success = false;
24439             }
24440         }
24441         var records = [];
24442         for(var i = 0; i < c; i++){
24443                 var n = root[i];
24444             var values = {};
24445             var id = this.getId(n);
24446             for(var j = 0; j < fl; j++){
24447                 f = fi[j];
24448             var v = this.ef[j](n);
24449             if (!f.convert) {
24450                 Roo.log('missing convert for ' + f.name);
24451                 Roo.log(f);
24452                 continue;
24453             }
24454             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24455             }
24456             var record = new Record(values, id);
24457             record.json = n;
24458             records[i] = record;
24459         }
24460         return {
24461             raw : o,
24462             success : success,
24463             records : records,
24464             totalRecords : totalRecords
24465         };
24466     }
24467 });/*
24468  * Based on:
24469  * Ext JS Library 1.1.1
24470  * Copyright(c) 2006-2007, Ext JS, LLC.
24471  *
24472  * Originally Released Under LGPL - original licence link has changed is not relivant.
24473  *
24474  * Fork - LGPL
24475  * <script type="text/javascript">
24476  */
24477
24478 /**
24479  * @class Roo.data.XmlReader
24480  * @extends Roo.data.DataReader
24481  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24482  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24483  * <p>
24484  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24485  * header in the HTTP response must be set to "text/xml".</em>
24486  * <p>
24487  * Example code:
24488  * <pre><code>
24489 var RecordDef = Roo.data.Record.create([
24490    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24491    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24492 ]);
24493 var myReader = new Roo.data.XmlReader({
24494    totalRecords: "results", // The element which contains the total dataset size (optional)
24495    record: "row",           // The repeated element which contains row information
24496    id: "id"                 // The element within the row that provides an ID for the record (optional)
24497 }, RecordDef);
24498 </code></pre>
24499  * <p>
24500  * This would consume an XML file like this:
24501  * <pre><code>
24502 &lt;?xml?>
24503 &lt;dataset>
24504  &lt;results>2&lt;/results>
24505  &lt;row>
24506    &lt;id>1&lt;/id>
24507    &lt;name>Bill&lt;/name>
24508    &lt;occupation>Gardener&lt;/occupation>
24509  &lt;/row>
24510  &lt;row>
24511    &lt;id>2&lt;/id>
24512    &lt;name>Ben&lt;/name>
24513    &lt;occupation>Horticulturalist&lt;/occupation>
24514  &lt;/row>
24515 &lt;/dataset>
24516 </code></pre>
24517  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24518  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24519  * paged from the remote server.
24520  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24521  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24522  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24523  * a record identifier value.
24524  * @constructor
24525  * Create a new XmlReader
24526  * @param {Object} meta Metadata configuration options
24527  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24528  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24529  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24530  */
24531 Roo.data.XmlReader = function(meta, recordType){
24532     meta = meta || {};
24533     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24534 };
24535 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24536     
24537     readerType : 'Xml',
24538     
24539     /**
24540      * This method is only used by a DataProxy which has retrieved data from a remote server.
24541          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24542          * to contain a method called 'responseXML' that returns an XML document object.
24543      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24544      * a cache of Roo.data.Records.
24545      */
24546     read : function(response){
24547         var doc = response.responseXML;
24548         if(!doc) {
24549             throw {message: "XmlReader.read: XML Document not available"};
24550         }
24551         return this.readRecords(doc);
24552     },
24553
24554     /**
24555      * Create a data block containing Roo.data.Records from an XML document.
24556          * @param {Object} doc A parsed XML document.
24557      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24558      * a cache of Roo.data.Records.
24559      */
24560     readRecords : function(doc){
24561         /**
24562          * After any data loads/reads, the raw XML Document is available for further custom processing.
24563          * @type XMLDocument
24564          */
24565         this.xmlData = doc;
24566         var root = doc.documentElement || doc;
24567         var q = Roo.DomQuery;
24568         var recordType = this.recordType, fields = recordType.prototype.fields;
24569         var sid = this.meta.id;
24570         var totalRecords = 0, success = true;
24571         if(this.meta.totalRecords){
24572             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24573         }
24574         
24575         if(this.meta.success){
24576             var sv = q.selectValue(this.meta.success, root, true);
24577             success = sv !== false && sv !== 'false';
24578         }
24579         var records = [];
24580         var ns = q.select(this.meta.record, root);
24581         for(var i = 0, len = ns.length; i < len; i++) {
24582                 var n = ns[i];
24583                 var values = {};
24584                 var id = sid ? q.selectValue(sid, n) : undefined;
24585                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24586                     var f = fields.items[j];
24587                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24588                     v = f.convert(v);
24589                     values[f.name] = v;
24590                 }
24591                 var record = new recordType(values, id);
24592                 record.node = n;
24593                 records[records.length] = record;
24594             }
24595
24596             return {
24597                 success : success,
24598                 records : records,
24599                 totalRecords : totalRecords || records.length
24600             };
24601     }
24602 });/*
24603  * Based on:
24604  * Ext JS Library 1.1.1
24605  * Copyright(c) 2006-2007, Ext JS, LLC.
24606  *
24607  * Originally Released Under LGPL - original licence link has changed is not relivant.
24608  *
24609  * Fork - LGPL
24610  * <script type="text/javascript">
24611  */
24612
24613 /**
24614  * @class Roo.data.ArrayReader
24615  * @extends Roo.data.DataReader
24616  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24617  * Each element of that Array represents a row of data fields. The
24618  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24619  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24620  * <p>
24621  * Example code:.
24622  * <pre><code>
24623 var RecordDef = Roo.data.Record.create([
24624     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24625     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24626 ]);
24627 var myReader = new Roo.data.ArrayReader({
24628     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24629 }, RecordDef);
24630 </code></pre>
24631  * <p>
24632  * This would consume an Array like this:
24633  * <pre><code>
24634 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24635   </code></pre>
24636  
24637  * @constructor
24638  * Create a new JsonReader
24639  * @param {Object} meta Metadata configuration options.
24640  * @param {Object|Array} recordType Either an Array of field definition objects
24641  * 
24642  * @cfg {Array} fields Array of field definition objects
24643  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24644  * as specified to {@link Roo.data.Record#create},
24645  * or an {@link Roo.data.Record} object
24646  *
24647  * 
24648  * created using {@link Roo.data.Record#create}.
24649  */
24650 Roo.data.ArrayReader = function(meta, recordType)
24651 {    
24652     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24653 };
24654
24655 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24656     
24657     readerType : 'Array',
24658     /**
24659      * Create a data block containing Roo.data.Records from an XML document.
24660      * @param {Object} o An Array of row objects which represents the dataset.
24661      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24662      * a cache of Roo.data.Records.
24663      */
24664     readRecords : function(o)
24665     {
24666         var sid = this.meta ? this.meta.id : null;
24667         var recordType = this.recordType, fields = recordType.prototype.fields;
24668         var records = [];
24669         var root = o;
24670         for(var i = 0; i < root.length; i++){
24671                 var n = root[i];
24672             var values = {};
24673             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24674             for(var j = 0, jlen = fields.length; j < jlen; j++){
24675                 var f = fields.items[j];
24676                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24677                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24678                 v = f.convert(v);
24679                 values[f.name] = v;
24680             }
24681             var record = new recordType(values, id);
24682             record.json = n;
24683             records[records.length] = record;
24684         }
24685         return {
24686             records : records,
24687             totalRecords : records.length
24688         };
24689     }
24690 });/*
24691  * Based on:
24692  * Ext JS Library 1.1.1
24693  * Copyright(c) 2006-2007, Ext JS, LLC.
24694  *
24695  * Originally Released Under LGPL - original licence link has changed is not relivant.
24696  *
24697  * Fork - LGPL
24698  * <script type="text/javascript">
24699  */
24700
24701
24702 /**
24703  * @class Roo.data.Tree
24704  * @extends Roo.util.Observable
24705  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24706  * in the tree have most standard DOM functionality.
24707  * @constructor
24708  * @param {Node} root (optional) The root node
24709  */
24710 Roo.data.Tree = function(root){
24711    this.nodeHash = {};
24712    /**
24713     * The root node for this tree
24714     * @type Node
24715     */
24716    this.root = null;
24717    if(root){
24718        this.setRootNode(root);
24719    }
24720    this.addEvents({
24721        /**
24722         * @event append
24723         * Fires when a new child node is appended to a node in this tree.
24724         * @param {Tree} tree The owner tree
24725         * @param {Node} parent The parent node
24726         * @param {Node} node The newly appended node
24727         * @param {Number} index The index of the newly appended node
24728         */
24729        "append" : true,
24730        /**
24731         * @event remove
24732         * Fires when a child node is removed from a node in this tree.
24733         * @param {Tree} tree The owner tree
24734         * @param {Node} parent The parent node
24735         * @param {Node} node The child node removed
24736         */
24737        "remove" : true,
24738        /**
24739         * @event move
24740         * Fires when a node is moved to a new location in the tree
24741         * @param {Tree} tree The owner tree
24742         * @param {Node} node The node moved
24743         * @param {Node} oldParent The old parent of this node
24744         * @param {Node} newParent The new parent of this node
24745         * @param {Number} index The index it was moved to
24746         */
24747        "move" : true,
24748        /**
24749         * @event insert
24750         * Fires when a new child node is inserted in a node in this tree.
24751         * @param {Tree} tree The owner tree
24752         * @param {Node} parent The parent node
24753         * @param {Node} node The child node inserted
24754         * @param {Node} refNode The child node the node was inserted before
24755         */
24756        "insert" : true,
24757        /**
24758         * @event beforeappend
24759         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24760         * @param {Tree} tree The owner tree
24761         * @param {Node} parent The parent node
24762         * @param {Node} node The child node to be appended
24763         */
24764        "beforeappend" : true,
24765        /**
24766         * @event beforeremove
24767         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24768         * @param {Tree} tree The owner tree
24769         * @param {Node} parent The parent node
24770         * @param {Node} node The child node to be removed
24771         */
24772        "beforeremove" : true,
24773        /**
24774         * @event beforemove
24775         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24776         * @param {Tree} tree The owner tree
24777         * @param {Node} node The node being moved
24778         * @param {Node} oldParent The parent of the node
24779         * @param {Node} newParent The new parent the node is moving to
24780         * @param {Number} index The index it is being moved to
24781         */
24782        "beforemove" : true,
24783        /**
24784         * @event beforeinsert
24785         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24786         * @param {Tree} tree The owner tree
24787         * @param {Node} parent The parent node
24788         * @param {Node} node The child node to be inserted
24789         * @param {Node} refNode The child node the node is being inserted before
24790         */
24791        "beforeinsert" : true
24792    });
24793
24794     Roo.data.Tree.superclass.constructor.call(this);
24795 };
24796
24797 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24798     pathSeparator: "/",
24799
24800     proxyNodeEvent : function(){
24801         return this.fireEvent.apply(this, arguments);
24802     },
24803
24804     /**
24805      * Returns the root node for this tree.
24806      * @return {Node}
24807      */
24808     getRootNode : function(){
24809         return this.root;
24810     },
24811
24812     /**
24813      * Sets the root node for this tree.
24814      * @param {Node} node
24815      * @return {Node}
24816      */
24817     setRootNode : function(node){
24818         this.root = node;
24819         node.ownerTree = this;
24820         node.isRoot = true;
24821         this.registerNode(node);
24822         return node;
24823     },
24824
24825     /**
24826      * Gets a node in this tree by its id.
24827      * @param {String} id
24828      * @return {Node}
24829      */
24830     getNodeById : function(id){
24831         return this.nodeHash[id];
24832     },
24833
24834     registerNode : function(node){
24835         this.nodeHash[node.id] = node;
24836     },
24837
24838     unregisterNode : function(node){
24839         delete this.nodeHash[node.id];
24840     },
24841
24842     toString : function(){
24843         return "[Tree"+(this.id?" "+this.id:"")+"]";
24844     }
24845 });
24846
24847 /**
24848  * @class Roo.data.Node
24849  * @extends Roo.util.Observable
24850  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24851  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24852  * @constructor
24853  * @param {Object} attributes The attributes/config for the node
24854  */
24855 Roo.data.Node = function(attributes){
24856     /**
24857      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24858      * @type {Object}
24859      */
24860     this.attributes = attributes || {};
24861     this.leaf = this.attributes.leaf;
24862     /**
24863      * The node id. @type String
24864      */
24865     this.id = this.attributes.id;
24866     if(!this.id){
24867         this.id = Roo.id(null, "ynode-");
24868         this.attributes.id = this.id;
24869     }
24870      
24871     
24872     /**
24873      * All child nodes of this node. @type Array
24874      */
24875     this.childNodes = [];
24876     if(!this.childNodes.indexOf){ // indexOf is a must
24877         this.childNodes.indexOf = function(o){
24878             for(var i = 0, len = this.length; i < len; i++){
24879                 if(this[i] == o) {
24880                     return i;
24881                 }
24882             }
24883             return -1;
24884         };
24885     }
24886     /**
24887      * The parent node for this node. @type Node
24888      */
24889     this.parentNode = null;
24890     /**
24891      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24892      */
24893     this.firstChild = null;
24894     /**
24895      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24896      */
24897     this.lastChild = null;
24898     /**
24899      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24900      */
24901     this.previousSibling = null;
24902     /**
24903      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24904      */
24905     this.nextSibling = null;
24906
24907     this.addEvents({
24908        /**
24909         * @event append
24910         * Fires when a new child node is appended
24911         * @param {Tree} tree The owner tree
24912         * @param {Node} this This node
24913         * @param {Node} node The newly appended node
24914         * @param {Number} index The index of the newly appended node
24915         */
24916        "append" : true,
24917        /**
24918         * @event remove
24919         * Fires when a child node is removed
24920         * @param {Tree} tree The owner tree
24921         * @param {Node} this This node
24922         * @param {Node} node The removed node
24923         */
24924        "remove" : true,
24925        /**
24926         * @event move
24927         * Fires when this node is moved to a new location in the tree
24928         * @param {Tree} tree The owner tree
24929         * @param {Node} this This node
24930         * @param {Node} oldParent The old parent of this node
24931         * @param {Node} newParent The new parent of this node
24932         * @param {Number} index The index it was moved to
24933         */
24934        "move" : true,
24935        /**
24936         * @event insert
24937         * Fires when a new child node is inserted.
24938         * @param {Tree} tree The owner tree
24939         * @param {Node} this This node
24940         * @param {Node} node The child node inserted
24941         * @param {Node} refNode The child node the node was inserted before
24942         */
24943        "insert" : true,
24944        /**
24945         * @event beforeappend
24946         * Fires before a new child is appended, return false to cancel the append.
24947         * @param {Tree} tree The owner tree
24948         * @param {Node} this This node
24949         * @param {Node} node The child node to be appended
24950         */
24951        "beforeappend" : true,
24952        /**
24953         * @event beforeremove
24954         * Fires before a child is removed, return false to cancel the remove.
24955         * @param {Tree} tree The owner tree
24956         * @param {Node} this This node
24957         * @param {Node} node The child node to be removed
24958         */
24959        "beforeremove" : true,
24960        /**
24961         * @event beforemove
24962         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24963         * @param {Tree} tree The owner tree
24964         * @param {Node} this This node
24965         * @param {Node} oldParent The parent of this node
24966         * @param {Node} newParent The new parent this node is moving to
24967         * @param {Number} index The index it is being moved to
24968         */
24969        "beforemove" : true,
24970        /**
24971         * @event beforeinsert
24972         * Fires before a new child is inserted, return false to cancel the insert.
24973         * @param {Tree} tree The owner tree
24974         * @param {Node} this This node
24975         * @param {Node} node The child node to be inserted
24976         * @param {Node} refNode The child node the node is being inserted before
24977         */
24978        "beforeinsert" : true
24979    });
24980     this.listeners = this.attributes.listeners;
24981     Roo.data.Node.superclass.constructor.call(this);
24982 };
24983
24984 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24985     fireEvent : function(evtName){
24986         // first do standard event for this node
24987         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24988             return false;
24989         }
24990         // then bubble it up to the tree if the event wasn't cancelled
24991         var ot = this.getOwnerTree();
24992         if(ot){
24993             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24994                 return false;
24995             }
24996         }
24997         return true;
24998     },
24999
25000     /**
25001      * Returns true if this node is a leaf
25002      * @return {Boolean}
25003      */
25004     isLeaf : function(){
25005         return this.leaf === true;
25006     },
25007
25008     // private
25009     setFirstChild : function(node){
25010         this.firstChild = node;
25011     },
25012
25013     //private
25014     setLastChild : function(node){
25015         this.lastChild = node;
25016     },
25017
25018
25019     /**
25020      * Returns true if this node is the last child of its parent
25021      * @return {Boolean}
25022      */
25023     isLast : function(){
25024        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25025     },
25026
25027     /**
25028      * Returns true if this node is the first child of its parent
25029      * @return {Boolean}
25030      */
25031     isFirst : function(){
25032        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25033     },
25034
25035     hasChildNodes : function(){
25036         return !this.isLeaf() && this.childNodes.length > 0;
25037     },
25038
25039     /**
25040      * Insert node(s) as the last child node of this node.
25041      * @param {Node/Array} node The node or Array of nodes to append
25042      * @return {Node} The appended node if single append, or null if an array was passed
25043      */
25044     appendChild : function(node){
25045         var multi = false;
25046         if(node instanceof Array){
25047             multi = node;
25048         }else if(arguments.length > 1){
25049             multi = arguments;
25050         }
25051         
25052         // if passed an array or multiple args do them one by one
25053         if(multi){
25054             for(var i = 0, len = multi.length; i < len; i++) {
25055                 this.appendChild(multi[i]);
25056             }
25057         }else{
25058             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25059                 return false;
25060             }
25061             var index = this.childNodes.length;
25062             var oldParent = node.parentNode;
25063             // it's a move, make sure we move it cleanly
25064             if(oldParent){
25065                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25066                     return false;
25067                 }
25068                 oldParent.removeChild(node);
25069             }
25070             
25071             index = this.childNodes.length;
25072             if(index == 0){
25073                 this.setFirstChild(node);
25074             }
25075             this.childNodes.push(node);
25076             node.parentNode = this;
25077             var ps = this.childNodes[index-1];
25078             if(ps){
25079                 node.previousSibling = ps;
25080                 ps.nextSibling = node;
25081             }else{
25082                 node.previousSibling = null;
25083             }
25084             node.nextSibling = null;
25085             this.setLastChild(node);
25086             node.setOwnerTree(this.getOwnerTree());
25087             this.fireEvent("append", this.ownerTree, this, node, index);
25088             if(this.ownerTree) {
25089                 this.ownerTree.fireEvent("appendnode", this, node, index);
25090             }
25091             if(oldParent){
25092                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25093             }
25094             return node;
25095         }
25096     },
25097
25098     /**
25099      * Removes a child node from this node.
25100      * @param {Node} node The node to remove
25101      * @return {Node} The removed node
25102      */
25103     removeChild : function(node){
25104         var index = this.childNodes.indexOf(node);
25105         if(index == -1){
25106             return false;
25107         }
25108         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25109             return false;
25110         }
25111
25112         // remove it from childNodes collection
25113         this.childNodes.splice(index, 1);
25114
25115         // update siblings
25116         if(node.previousSibling){
25117             node.previousSibling.nextSibling = node.nextSibling;
25118         }
25119         if(node.nextSibling){
25120             node.nextSibling.previousSibling = node.previousSibling;
25121         }
25122
25123         // update child refs
25124         if(this.firstChild == node){
25125             this.setFirstChild(node.nextSibling);
25126         }
25127         if(this.lastChild == node){
25128             this.setLastChild(node.previousSibling);
25129         }
25130
25131         node.setOwnerTree(null);
25132         // clear any references from the node
25133         node.parentNode = null;
25134         node.previousSibling = null;
25135         node.nextSibling = null;
25136         this.fireEvent("remove", this.ownerTree, this, node);
25137         return node;
25138     },
25139
25140     /**
25141      * Inserts the first node before the second node in this nodes childNodes collection.
25142      * @param {Node} node The node to insert
25143      * @param {Node} refNode The node to insert before (if null the node is appended)
25144      * @return {Node} The inserted node
25145      */
25146     insertBefore : function(node, refNode){
25147         if(!refNode){ // like standard Dom, refNode can be null for append
25148             return this.appendChild(node);
25149         }
25150         // nothing to do
25151         if(node == refNode){
25152             return false;
25153         }
25154
25155         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25156             return false;
25157         }
25158         var index = this.childNodes.indexOf(refNode);
25159         var oldParent = node.parentNode;
25160         var refIndex = index;
25161
25162         // when moving internally, indexes will change after remove
25163         if(oldParent == this && this.childNodes.indexOf(node) < index){
25164             refIndex--;
25165         }
25166
25167         // it's a move, make sure we move it cleanly
25168         if(oldParent){
25169             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25170                 return false;
25171             }
25172             oldParent.removeChild(node);
25173         }
25174         if(refIndex == 0){
25175             this.setFirstChild(node);
25176         }
25177         this.childNodes.splice(refIndex, 0, node);
25178         node.parentNode = this;
25179         var ps = this.childNodes[refIndex-1];
25180         if(ps){
25181             node.previousSibling = ps;
25182             ps.nextSibling = node;
25183         }else{
25184             node.previousSibling = null;
25185         }
25186         node.nextSibling = refNode;
25187         refNode.previousSibling = node;
25188         node.setOwnerTree(this.getOwnerTree());
25189         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25190         if(oldParent){
25191             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25192         }
25193         return node;
25194     },
25195
25196     /**
25197      * Returns the child node at the specified index.
25198      * @param {Number} index
25199      * @return {Node}
25200      */
25201     item : function(index){
25202         return this.childNodes[index];
25203     },
25204
25205     /**
25206      * Replaces one child node in this node with another.
25207      * @param {Node} newChild The replacement node
25208      * @param {Node} oldChild The node to replace
25209      * @return {Node} The replaced node
25210      */
25211     replaceChild : function(newChild, oldChild){
25212         this.insertBefore(newChild, oldChild);
25213         this.removeChild(oldChild);
25214         return oldChild;
25215     },
25216
25217     /**
25218      * Returns the index of a child node
25219      * @param {Node} node
25220      * @return {Number} The index of the node or -1 if it was not found
25221      */
25222     indexOf : function(child){
25223         return this.childNodes.indexOf(child);
25224     },
25225
25226     /**
25227      * Returns the tree this node is in.
25228      * @return {Tree}
25229      */
25230     getOwnerTree : function(){
25231         // if it doesn't have one, look for one
25232         if(!this.ownerTree){
25233             var p = this;
25234             while(p){
25235                 if(p.ownerTree){
25236                     this.ownerTree = p.ownerTree;
25237                     break;
25238                 }
25239                 p = p.parentNode;
25240             }
25241         }
25242         return this.ownerTree;
25243     },
25244
25245     /**
25246      * Returns depth of this node (the root node has a depth of 0)
25247      * @return {Number}
25248      */
25249     getDepth : function(){
25250         var depth = 0;
25251         var p = this;
25252         while(p.parentNode){
25253             ++depth;
25254             p = p.parentNode;
25255         }
25256         return depth;
25257     },
25258
25259     // private
25260     setOwnerTree : function(tree){
25261         // if it's move, we need to update everyone
25262         if(tree != this.ownerTree){
25263             if(this.ownerTree){
25264                 this.ownerTree.unregisterNode(this);
25265             }
25266             this.ownerTree = tree;
25267             var cs = this.childNodes;
25268             for(var i = 0, len = cs.length; i < len; i++) {
25269                 cs[i].setOwnerTree(tree);
25270             }
25271             if(tree){
25272                 tree.registerNode(this);
25273             }
25274         }
25275     },
25276
25277     /**
25278      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25279      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25280      * @return {String} The path
25281      */
25282     getPath : function(attr){
25283         attr = attr || "id";
25284         var p = this.parentNode;
25285         var b = [this.attributes[attr]];
25286         while(p){
25287             b.unshift(p.attributes[attr]);
25288             p = p.parentNode;
25289         }
25290         var sep = this.getOwnerTree().pathSeparator;
25291         return sep + b.join(sep);
25292     },
25293
25294     /**
25295      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25296      * function call will be the scope provided or the current node. The arguments to the function
25297      * will be the args provided or the current node. If the function returns false at any point,
25298      * the bubble is stopped.
25299      * @param {Function} fn The function to call
25300      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25301      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25302      */
25303     bubble : function(fn, scope, args){
25304         var p = this;
25305         while(p){
25306             if(fn.call(scope || p, args || p) === false){
25307                 break;
25308             }
25309             p = p.parentNode;
25310         }
25311     },
25312
25313     /**
25314      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25315      * function call will be the scope provided or the current node. The arguments to the function
25316      * will be the args provided or the current node. If the function returns false at any point,
25317      * the cascade is stopped on that branch.
25318      * @param {Function} fn The function to call
25319      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25320      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25321      */
25322     cascade : function(fn, scope, args){
25323         if(fn.call(scope || this, args || this) !== false){
25324             var cs = this.childNodes;
25325             for(var i = 0, len = cs.length; i < len; i++) {
25326                 cs[i].cascade(fn, scope, args);
25327             }
25328         }
25329     },
25330
25331     /**
25332      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25333      * function call will be the scope provided or the current node. The arguments to the function
25334      * will be the args provided or the current node. If the function returns false at any point,
25335      * the iteration stops.
25336      * @param {Function} fn The function to call
25337      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25338      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25339      */
25340     eachChild : function(fn, scope, args){
25341         var cs = this.childNodes;
25342         for(var i = 0, len = cs.length; i < len; i++) {
25343                 if(fn.call(scope || this, args || cs[i]) === false){
25344                     break;
25345                 }
25346         }
25347     },
25348
25349     /**
25350      * Finds the first child that has the attribute with the specified value.
25351      * @param {String} attribute The attribute name
25352      * @param {Mixed} value The value to search for
25353      * @return {Node} The found child or null if none was found
25354      */
25355     findChild : function(attribute, value){
25356         var cs = this.childNodes;
25357         for(var i = 0, len = cs.length; i < len; i++) {
25358                 if(cs[i].attributes[attribute] == value){
25359                     return cs[i];
25360                 }
25361         }
25362         return null;
25363     },
25364
25365     /**
25366      * Finds the first child by a custom function. The child matches if the function passed
25367      * returns true.
25368      * @param {Function} fn
25369      * @param {Object} scope (optional)
25370      * @return {Node} The found child or null if none was found
25371      */
25372     findChildBy : function(fn, scope){
25373         var cs = this.childNodes;
25374         for(var i = 0, len = cs.length; i < len; i++) {
25375                 if(fn.call(scope||cs[i], cs[i]) === true){
25376                     return cs[i];
25377                 }
25378         }
25379         return null;
25380     },
25381
25382     /**
25383      * Sorts this nodes children using the supplied sort function
25384      * @param {Function} fn
25385      * @param {Object} scope (optional)
25386      */
25387     sort : function(fn, scope){
25388         var cs = this.childNodes;
25389         var len = cs.length;
25390         if(len > 0){
25391             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25392             cs.sort(sortFn);
25393             for(var i = 0; i < len; i++){
25394                 var n = cs[i];
25395                 n.previousSibling = cs[i-1];
25396                 n.nextSibling = cs[i+1];
25397                 if(i == 0){
25398                     this.setFirstChild(n);
25399                 }
25400                 if(i == len-1){
25401                     this.setLastChild(n);
25402                 }
25403             }
25404         }
25405     },
25406
25407     /**
25408      * Returns true if this node is an ancestor (at any point) of the passed node.
25409      * @param {Node} node
25410      * @return {Boolean}
25411      */
25412     contains : function(node){
25413         return node.isAncestor(this);
25414     },
25415
25416     /**
25417      * Returns true if the passed node is an ancestor (at any point) of this node.
25418      * @param {Node} node
25419      * @return {Boolean}
25420      */
25421     isAncestor : function(node){
25422         var p = this.parentNode;
25423         while(p){
25424             if(p == node){
25425                 return true;
25426             }
25427             p = p.parentNode;
25428         }
25429         return false;
25430     },
25431
25432     toString : function(){
25433         return "[Node"+(this.id?" "+this.id:"")+"]";
25434     }
25435 });/*
25436  * Based on:
25437  * Ext JS Library 1.1.1
25438  * Copyright(c) 2006-2007, Ext JS, LLC.
25439  *
25440  * Originally Released Under LGPL - original licence link has changed is not relivant.
25441  *
25442  * Fork - LGPL
25443  * <script type="text/javascript">
25444  */
25445  (function(){ 
25446 /**
25447  * @class Roo.Layer
25448  * @extends Roo.Element
25449  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25450  * automatic maintaining of shadow/shim positions.
25451  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25452  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25453  * you can pass a string with a CSS class name. False turns off the shadow.
25454  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25455  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25456  * @cfg {String} cls CSS class to add to the element
25457  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25458  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25459  * @constructor
25460  * @param {Object} config An object with config options.
25461  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25462  */
25463
25464 Roo.Layer = function(config, existingEl){
25465     config = config || {};
25466     var dh = Roo.DomHelper;
25467     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25468     if(existingEl){
25469         this.dom = Roo.getDom(existingEl);
25470     }
25471     if(!this.dom){
25472         var o = config.dh || {tag: "div", cls: "x-layer"};
25473         this.dom = dh.append(pel, o);
25474     }
25475     if(config.cls){
25476         this.addClass(config.cls);
25477     }
25478     this.constrain = config.constrain !== false;
25479     this.visibilityMode = Roo.Element.VISIBILITY;
25480     if(config.id){
25481         this.id = this.dom.id = config.id;
25482     }else{
25483         this.id = Roo.id(this.dom);
25484     }
25485     this.zindex = config.zindex || this.getZIndex();
25486     this.position("absolute", this.zindex);
25487     if(config.shadow){
25488         this.shadowOffset = config.shadowOffset || 4;
25489         this.shadow = new Roo.Shadow({
25490             offset : this.shadowOffset,
25491             mode : config.shadow
25492         });
25493     }else{
25494         this.shadowOffset = 0;
25495     }
25496     this.useShim = config.shim !== false && Roo.useShims;
25497     this.useDisplay = config.useDisplay;
25498     this.hide();
25499 };
25500
25501 var supr = Roo.Element.prototype;
25502
25503 // shims are shared among layer to keep from having 100 iframes
25504 var shims = [];
25505
25506 Roo.extend(Roo.Layer, Roo.Element, {
25507
25508     getZIndex : function(){
25509         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25510     },
25511
25512     getShim : function(){
25513         if(!this.useShim){
25514             return null;
25515         }
25516         if(this.shim){
25517             return this.shim;
25518         }
25519         var shim = shims.shift();
25520         if(!shim){
25521             shim = this.createShim();
25522             shim.enableDisplayMode('block');
25523             shim.dom.style.display = 'none';
25524             shim.dom.style.visibility = 'visible';
25525         }
25526         var pn = this.dom.parentNode;
25527         if(shim.dom.parentNode != pn){
25528             pn.insertBefore(shim.dom, this.dom);
25529         }
25530         shim.setStyle('z-index', this.getZIndex()-2);
25531         this.shim = shim;
25532         return shim;
25533     },
25534
25535     hideShim : function(){
25536         if(this.shim){
25537             this.shim.setDisplayed(false);
25538             shims.push(this.shim);
25539             delete this.shim;
25540         }
25541     },
25542
25543     disableShadow : function(){
25544         if(this.shadow){
25545             this.shadowDisabled = true;
25546             this.shadow.hide();
25547             this.lastShadowOffset = this.shadowOffset;
25548             this.shadowOffset = 0;
25549         }
25550     },
25551
25552     enableShadow : function(show){
25553         if(this.shadow){
25554             this.shadowDisabled = false;
25555             this.shadowOffset = this.lastShadowOffset;
25556             delete this.lastShadowOffset;
25557             if(show){
25558                 this.sync(true);
25559             }
25560         }
25561     },
25562
25563     // private
25564     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25565     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25566     sync : function(doShow){
25567         var sw = this.shadow;
25568         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25569             var sh = this.getShim();
25570
25571             var w = this.getWidth(),
25572                 h = this.getHeight();
25573
25574             var l = this.getLeft(true),
25575                 t = this.getTop(true);
25576
25577             if(sw && !this.shadowDisabled){
25578                 if(doShow && !sw.isVisible()){
25579                     sw.show(this);
25580                 }else{
25581                     sw.realign(l, t, w, h);
25582                 }
25583                 if(sh){
25584                     if(doShow){
25585                        sh.show();
25586                     }
25587                     // fit the shim behind the shadow, so it is shimmed too
25588                     var a = sw.adjusts, s = sh.dom.style;
25589                     s.left = (Math.min(l, l+a.l))+"px";
25590                     s.top = (Math.min(t, t+a.t))+"px";
25591                     s.width = (w+a.w)+"px";
25592                     s.height = (h+a.h)+"px";
25593                 }
25594             }else if(sh){
25595                 if(doShow){
25596                    sh.show();
25597                 }
25598                 sh.setSize(w, h);
25599                 sh.setLeftTop(l, t);
25600             }
25601             
25602         }
25603     },
25604
25605     // private
25606     destroy : function(){
25607         this.hideShim();
25608         if(this.shadow){
25609             this.shadow.hide();
25610         }
25611         this.removeAllListeners();
25612         var pn = this.dom.parentNode;
25613         if(pn){
25614             pn.removeChild(this.dom);
25615         }
25616         Roo.Element.uncache(this.id);
25617     },
25618
25619     remove : function(){
25620         this.destroy();
25621     },
25622
25623     // private
25624     beginUpdate : function(){
25625         this.updating = true;
25626     },
25627
25628     // private
25629     endUpdate : function(){
25630         this.updating = false;
25631         this.sync(true);
25632     },
25633
25634     // private
25635     hideUnders : function(negOffset){
25636         if(this.shadow){
25637             this.shadow.hide();
25638         }
25639         this.hideShim();
25640     },
25641
25642     // private
25643     constrainXY : function(){
25644         if(this.constrain){
25645             var vw = Roo.lib.Dom.getViewWidth(),
25646                 vh = Roo.lib.Dom.getViewHeight();
25647             var s = Roo.get(document).getScroll();
25648
25649             var xy = this.getXY();
25650             var x = xy[0], y = xy[1];   
25651             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25652             // only move it if it needs it
25653             var moved = false;
25654             // first validate right/bottom
25655             if((x + w) > vw+s.left){
25656                 x = vw - w - this.shadowOffset;
25657                 moved = true;
25658             }
25659             if((y + h) > vh+s.top){
25660                 y = vh - h - this.shadowOffset;
25661                 moved = true;
25662             }
25663             // then make sure top/left isn't negative
25664             if(x < s.left){
25665                 x = s.left;
25666                 moved = true;
25667             }
25668             if(y < s.top){
25669                 y = s.top;
25670                 moved = true;
25671             }
25672             if(moved){
25673                 if(this.avoidY){
25674                     var ay = this.avoidY;
25675                     if(y <= ay && (y+h) >= ay){
25676                         y = ay-h-5;   
25677                     }
25678                 }
25679                 xy = [x, y];
25680                 this.storeXY(xy);
25681                 supr.setXY.call(this, xy);
25682                 this.sync();
25683             }
25684         }
25685     },
25686
25687     isVisible : function(){
25688         return this.visible;    
25689     },
25690
25691     // private
25692     showAction : function(){
25693         this.visible = true; // track visibility to prevent getStyle calls
25694         if(this.useDisplay === true){
25695             this.setDisplayed("");
25696         }else if(this.lastXY){
25697             supr.setXY.call(this, this.lastXY);
25698         }else if(this.lastLT){
25699             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25700         }
25701     },
25702
25703     // private
25704     hideAction : function(){
25705         this.visible = false;
25706         if(this.useDisplay === true){
25707             this.setDisplayed(false);
25708         }else{
25709             this.setLeftTop(-10000,-10000);
25710         }
25711     },
25712
25713     // overridden Element method
25714     setVisible : function(v, a, d, c, e){
25715         if(v){
25716             this.showAction();
25717         }
25718         if(a && v){
25719             var cb = function(){
25720                 this.sync(true);
25721                 if(c){
25722                     c();
25723                 }
25724             }.createDelegate(this);
25725             supr.setVisible.call(this, true, true, d, cb, e);
25726         }else{
25727             if(!v){
25728                 this.hideUnders(true);
25729             }
25730             var cb = c;
25731             if(a){
25732                 cb = function(){
25733                     this.hideAction();
25734                     if(c){
25735                         c();
25736                     }
25737                 }.createDelegate(this);
25738             }
25739             supr.setVisible.call(this, v, a, d, cb, e);
25740             if(v){
25741                 this.sync(true);
25742             }else if(!a){
25743                 this.hideAction();
25744             }
25745         }
25746     },
25747
25748     storeXY : function(xy){
25749         delete this.lastLT;
25750         this.lastXY = xy;
25751     },
25752
25753     storeLeftTop : function(left, top){
25754         delete this.lastXY;
25755         this.lastLT = [left, top];
25756     },
25757
25758     // private
25759     beforeFx : function(){
25760         this.beforeAction();
25761         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25762     },
25763
25764     // private
25765     afterFx : function(){
25766         Roo.Layer.superclass.afterFx.apply(this, arguments);
25767         this.sync(this.isVisible());
25768     },
25769
25770     // private
25771     beforeAction : function(){
25772         if(!this.updating && this.shadow){
25773             this.shadow.hide();
25774         }
25775     },
25776
25777     // overridden Element method
25778     setLeft : function(left){
25779         this.storeLeftTop(left, this.getTop(true));
25780         supr.setLeft.apply(this, arguments);
25781         this.sync();
25782     },
25783
25784     setTop : function(top){
25785         this.storeLeftTop(this.getLeft(true), top);
25786         supr.setTop.apply(this, arguments);
25787         this.sync();
25788     },
25789
25790     setLeftTop : function(left, top){
25791         this.storeLeftTop(left, top);
25792         supr.setLeftTop.apply(this, arguments);
25793         this.sync();
25794     },
25795
25796     setXY : function(xy, a, d, c, e){
25797         this.fixDisplay();
25798         this.beforeAction();
25799         this.storeXY(xy);
25800         var cb = this.createCB(c);
25801         supr.setXY.call(this, xy, a, d, cb, e);
25802         if(!a){
25803             cb();
25804         }
25805     },
25806
25807     // private
25808     createCB : function(c){
25809         var el = this;
25810         return function(){
25811             el.constrainXY();
25812             el.sync(true);
25813             if(c){
25814                 c();
25815             }
25816         };
25817     },
25818
25819     // overridden Element method
25820     setX : function(x, a, d, c, e){
25821         this.setXY([x, this.getY()], a, d, c, e);
25822     },
25823
25824     // overridden Element method
25825     setY : function(y, a, d, c, e){
25826         this.setXY([this.getX(), y], a, d, c, e);
25827     },
25828
25829     // overridden Element method
25830     setSize : function(w, h, a, d, c, e){
25831         this.beforeAction();
25832         var cb = this.createCB(c);
25833         supr.setSize.call(this, w, h, a, d, cb, e);
25834         if(!a){
25835             cb();
25836         }
25837     },
25838
25839     // overridden Element method
25840     setWidth : function(w, a, d, c, e){
25841         this.beforeAction();
25842         var cb = this.createCB(c);
25843         supr.setWidth.call(this, w, a, d, cb, e);
25844         if(!a){
25845             cb();
25846         }
25847     },
25848
25849     // overridden Element method
25850     setHeight : function(h, a, d, c, e){
25851         this.beforeAction();
25852         var cb = this.createCB(c);
25853         supr.setHeight.call(this, h, a, d, cb, e);
25854         if(!a){
25855             cb();
25856         }
25857     },
25858
25859     // overridden Element method
25860     setBounds : function(x, y, w, h, a, d, c, e){
25861         this.beforeAction();
25862         var cb = this.createCB(c);
25863         if(!a){
25864             this.storeXY([x, y]);
25865             supr.setXY.call(this, [x, y]);
25866             supr.setSize.call(this, w, h, a, d, cb, e);
25867             cb();
25868         }else{
25869             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25870         }
25871         return this;
25872     },
25873     
25874     /**
25875      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25876      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25877      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25878      * @param {Number} zindex The new z-index to set
25879      * @return {this} The Layer
25880      */
25881     setZIndex : function(zindex){
25882         this.zindex = zindex;
25883         this.setStyle("z-index", zindex + 2);
25884         if(this.shadow){
25885             this.shadow.setZIndex(zindex + 1);
25886         }
25887         if(this.shim){
25888             this.shim.setStyle("z-index", zindex);
25889         }
25890     }
25891 });
25892 })();/*
25893  * Based on:
25894  * Ext JS Library 1.1.1
25895  * Copyright(c) 2006-2007, Ext JS, LLC.
25896  *
25897  * Originally Released Under LGPL - original licence link has changed is not relivant.
25898  *
25899  * Fork - LGPL
25900  * <script type="text/javascript">
25901  */
25902
25903
25904 /**
25905  * @class Roo.Shadow
25906  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25907  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25908  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25909  * @constructor
25910  * Create a new Shadow
25911  * @param {Object} config The config object
25912  */
25913 Roo.Shadow = function(config){
25914     Roo.apply(this, config);
25915     if(typeof this.mode != "string"){
25916         this.mode = this.defaultMode;
25917     }
25918     var o = this.offset, a = {h: 0};
25919     var rad = Math.floor(this.offset/2);
25920     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25921         case "drop":
25922             a.w = 0;
25923             a.l = a.t = o;
25924             a.t -= 1;
25925             if(Roo.isIE){
25926                 a.l -= this.offset + rad;
25927                 a.t -= this.offset + rad;
25928                 a.w -= rad;
25929                 a.h -= rad;
25930                 a.t += 1;
25931             }
25932         break;
25933         case "sides":
25934             a.w = (o*2);
25935             a.l = -o;
25936             a.t = o-1;
25937             if(Roo.isIE){
25938                 a.l -= (this.offset - rad);
25939                 a.t -= this.offset + rad;
25940                 a.l += 1;
25941                 a.w -= (this.offset - rad)*2;
25942                 a.w -= rad + 1;
25943                 a.h -= 1;
25944             }
25945         break;
25946         case "frame":
25947             a.w = a.h = (o*2);
25948             a.l = a.t = -o;
25949             a.t += 1;
25950             a.h -= 2;
25951             if(Roo.isIE){
25952                 a.l -= (this.offset - rad);
25953                 a.t -= (this.offset - rad);
25954                 a.l += 1;
25955                 a.w -= (this.offset + rad + 1);
25956                 a.h -= (this.offset + rad);
25957                 a.h += 1;
25958             }
25959         break;
25960     };
25961
25962     this.adjusts = a;
25963 };
25964
25965 Roo.Shadow.prototype = {
25966     /**
25967      * @cfg {String} mode
25968      * The shadow display mode.  Supports the following options:<br />
25969      * sides: Shadow displays on both sides and bottom only<br />
25970      * frame: Shadow displays equally on all four sides<br />
25971      * drop: Traditional bottom-right drop shadow (default)
25972      */
25973     /**
25974      * @cfg {String} offset
25975      * The number of pixels to offset the shadow from the element (defaults to 4)
25976      */
25977     offset: 4,
25978
25979     // private
25980     defaultMode: "drop",
25981
25982     /**
25983      * Displays the shadow under the target element
25984      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25985      */
25986     show : function(target){
25987         target = Roo.get(target);
25988         if(!this.el){
25989             this.el = Roo.Shadow.Pool.pull();
25990             if(this.el.dom.nextSibling != target.dom){
25991                 this.el.insertBefore(target);
25992             }
25993         }
25994         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25995         if(Roo.isIE){
25996             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25997         }
25998         this.realign(
25999             target.getLeft(true),
26000             target.getTop(true),
26001             target.getWidth(),
26002             target.getHeight()
26003         );
26004         this.el.dom.style.display = "block";
26005     },
26006
26007     /**
26008      * Returns true if the shadow is visible, else false
26009      */
26010     isVisible : function(){
26011         return this.el ? true : false;  
26012     },
26013
26014     /**
26015      * Direct alignment when values are already available. Show must be called at least once before
26016      * calling this method to ensure it is initialized.
26017      * @param {Number} left The target element left position
26018      * @param {Number} top The target element top position
26019      * @param {Number} width The target element width
26020      * @param {Number} height The target element height
26021      */
26022     realign : function(l, t, w, h){
26023         if(!this.el){
26024             return;
26025         }
26026         var a = this.adjusts, d = this.el.dom, s = d.style;
26027         var iea = 0;
26028         s.left = (l+a.l)+"px";
26029         s.top = (t+a.t)+"px";
26030         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26031  
26032         if(s.width != sws || s.height != shs){
26033             s.width = sws;
26034             s.height = shs;
26035             if(!Roo.isIE){
26036                 var cn = d.childNodes;
26037                 var sww = Math.max(0, (sw-12))+"px";
26038                 cn[0].childNodes[1].style.width = sww;
26039                 cn[1].childNodes[1].style.width = sww;
26040                 cn[2].childNodes[1].style.width = sww;
26041                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26042             }
26043         }
26044     },
26045
26046     /**
26047      * Hides this shadow
26048      */
26049     hide : function(){
26050         if(this.el){
26051             this.el.dom.style.display = "none";
26052             Roo.Shadow.Pool.push(this.el);
26053             delete this.el;
26054         }
26055     },
26056
26057     /**
26058      * Adjust the z-index of this shadow
26059      * @param {Number} zindex The new z-index
26060      */
26061     setZIndex : function(z){
26062         this.zIndex = z;
26063         if(this.el){
26064             this.el.setStyle("z-index", z);
26065         }
26066     }
26067 };
26068
26069 // Private utility class that manages the internal Shadow cache
26070 Roo.Shadow.Pool = function(){
26071     var p = [];
26072     var markup = Roo.isIE ?
26073                  '<div class="x-ie-shadow"></div>' :
26074                  '<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>';
26075     return {
26076         pull : function(){
26077             var sh = p.shift();
26078             if(!sh){
26079                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26080                 sh.autoBoxAdjust = false;
26081             }
26082             return sh;
26083         },
26084
26085         push : function(sh){
26086             p.push(sh);
26087         }
26088     };
26089 }();/*
26090  * Based on:
26091  * Ext JS Library 1.1.1
26092  * Copyright(c) 2006-2007, Ext JS, LLC.
26093  *
26094  * Originally Released Under LGPL - original licence link has changed is not relivant.
26095  *
26096  * Fork - LGPL
26097  * <script type="text/javascript">
26098  */
26099
26100
26101 /**
26102  * @class Roo.SplitBar
26103  * @extends Roo.util.Observable
26104  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26105  * <br><br>
26106  * Usage:
26107  * <pre><code>
26108 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26109                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26110 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26111 split.minSize = 100;
26112 split.maxSize = 600;
26113 split.animate = true;
26114 split.on('moved', splitterMoved);
26115 </code></pre>
26116  * @constructor
26117  * Create a new SplitBar
26118  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26119  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26120  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26121  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26122                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26123                         position of the SplitBar).
26124  */
26125 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26126     
26127     /** @private */
26128     this.el = Roo.get(dragElement, true);
26129     this.el.dom.unselectable = "on";
26130     /** @private */
26131     this.resizingEl = Roo.get(resizingElement, true);
26132
26133     /**
26134      * @private
26135      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26136      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26137      * @type Number
26138      */
26139     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26140     
26141     /**
26142      * The minimum size of the resizing element. (Defaults to 0)
26143      * @type Number
26144      */
26145     this.minSize = 0;
26146     
26147     /**
26148      * The maximum size of the resizing element. (Defaults to 2000)
26149      * @type Number
26150      */
26151     this.maxSize = 2000;
26152     
26153     /**
26154      * Whether to animate the transition to the new size
26155      * @type Boolean
26156      */
26157     this.animate = false;
26158     
26159     /**
26160      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26161      * @type Boolean
26162      */
26163     this.useShim = false;
26164     
26165     /** @private */
26166     this.shim = null;
26167     
26168     if(!existingProxy){
26169         /** @private */
26170         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26171     }else{
26172         this.proxy = Roo.get(existingProxy).dom;
26173     }
26174     /** @private */
26175     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26176     
26177     /** @private */
26178     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26179     
26180     /** @private */
26181     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26182     
26183     /** @private */
26184     this.dragSpecs = {};
26185     
26186     /**
26187      * @private The adapter to use to positon and resize elements
26188      */
26189     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26190     this.adapter.init(this);
26191     
26192     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26193         /** @private */
26194         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26195         this.el.addClass("x-splitbar-h");
26196     }else{
26197         /** @private */
26198         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26199         this.el.addClass("x-splitbar-v");
26200     }
26201     
26202     this.addEvents({
26203         /**
26204          * @event resize
26205          * Fires when the splitter is moved (alias for {@link #event-moved})
26206          * @param {Roo.SplitBar} this
26207          * @param {Number} newSize the new width or height
26208          */
26209         "resize" : true,
26210         /**
26211          * @event moved
26212          * Fires when the splitter is moved
26213          * @param {Roo.SplitBar} this
26214          * @param {Number} newSize the new width or height
26215          */
26216         "moved" : true,
26217         /**
26218          * @event beforeresize
26219          * Fires before the splitter is dragged
26220          * @param {Roo.SplitBar} this
26221          */
26222         "beforeresize" : true,
26223
26224         "beforeapply" : true
26225     });
26226
26227     Roo.util.Observable.call(this);
26228 };
26229
26230 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26231     onStartProxyDrag : function(x, y){
26232         this.fireEvent("beforeresize", this);
26233         if(!this.overlay){
26234             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26235             o.unselectable();
26236             o.enableDisplayMode("block");
26237             // all splitbars share the same overlay
26238             Roo.SplitBar.prototype.overlay = o;
26239         }
26240         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26241         this.overlay.show();
26242         Roo.get(this.proxy).setDisplayed("block");
26243         var size = this.adapter.getElementSize(this);
26244         this.activeMinSize = this.getMinimumSize();;
26245         this.activeMaxSize = this.getMaximumSize();;
26246         var c1 = size - this.activeMinSize;
26247         var c2 = Math.max(this.activeMaxSize - size, 0);
26248         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26249             this.dd.resetConstraints();
26250             this.dd.setXConstraint(
26251                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26252                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26253             );
26254             this.dd.setYConstraint(0, 0);
26255         }else{
26256             this.dd.resetConstraints();
26257             this.dd.setXConstraint(0, 0);
26258             this.dd.setYConstraint(
26259                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26260                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26261             );
26262          }
26263         this.dragSpecs.startSize = size;
26264         this.dragSpecs.startPoint = [x, y];
26265         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26266     },
26267     
26268     /** 
26269      * @private Called after the drag operation by the DDProxy
26270      */
26271     onEndProxyDrag : function(e){
26272         Roo.get(this.proxy).setDisplayed(false);
26273         var endPoint = Roo.lib.Event.getXY(e);
26274         if(this.overlay){
26275             this.overlay.hide();
26276         }
26277         var newSize;
26278         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26279             newSize = this.dragSpecs.startSize + 
26280                 (this.placement == Roo.SplitBar.LEFT ?
26281                     endPoint[0] - this.dragSpecs.startPoint[0] :
26282                     this.dragSpecs.startPoint[0] - endPoint[0]
26283                 );
26284         }else{
26285             newSize = this.dragSpecs.startSize + 
26286                 (this.placement == Roo.SplitBar.TOP ?
26287                     endPoint[1] - this.dragSpecs.startPoint[1] :
26288                     this.dragSpecs.startPoint[1] - endPoint[1]
26289                 );
26290         }
26291         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26292         if(newSize != this.dragSpecs.startSize){
26293             if(this.fireEvent('beforeapply', this, newSize) !== false){
26294                 this.adapter.setElementSize(this, newSize);
26295                 this.fireEvent("moved", this, newSize);
26296                 this.fireEvent("resize", this, newSize);
26297             }
26298         }
26299     },
26300     
26301     /**
26302      * Get the adapter this SplitBar uses
26303      * @return The adapter object
26304      */
26305     getAdapter : function(){
26306         return this.adapter;
26307     },
26308     
26309     /**
26310      * Set the adapter this SplitBar uses
26311      * @param {Object} adapter A SplitBar adapter object
26312      */
26313     setAdapter : function(adapter){
26314         this.adapter = adapter;
26315         this.adapter.init(this);
26316     },
26317     
26318     /**
26319      * Gets the minimum size for the resizing element
26320      * @return {Number} The minimum size
26321      */
26322     getMinimumSize : function(){
26323         return this.minSize;
26324     },
26325     
26326     /**
26327      * Sets the minimum size for the resizing element
26328      * @param {Number} minSize The minimum size
26329      */
26330     setMinimumSize : function(minSize){
26331         this.minSize = minSize;
26332     },
26333     
26334     /**
26335      * Gets the maximum size for the resizing element
26336      * @return {Number} The maximum size
26337      */
26338     getMaximumSize : function(){
26339         return this.maxSize;
26340     },
26341     
26342     /**
26343      * Sets the maximum size for the resizing element
26344      * @param {Number} maxSize The maximum size
26345      */
26346     setMaximumSize : function(maxSize){
26347         this.maxSize = maxSize;
26348     },
26349     
26350     /**
26351      * Sets the initialize size for the resizing element
26352      * @param {Number} size The initial size
26353      */
26354     setCurrentSize : function(size){
26355         var oldAnimate = this.animate;
26356         this.animate = false;
26357         this.adapter.setElementSize(this, size);
26358         this.animate = oldAnimate;
26359     },
26360     
26361     /**
26362      * Destroy this splitbar. 
26363      * @param {Boolean} removeEl True to remove the element
26364      */
26365     destroy : function(removeEl){
26366         if(this.shim){
26367             this.shim.remove();
26368         }
26369         this.dd.unreg();
26370         this.proxy.parentNode.removeChild(this.proxy);
26371         if(removeEl){
26372             this.el.remove();
26373         }
26374     }
26375 });
26376
26377 /**
26378  * @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.
26379  */
26380 Roo.SplitBar.createProxy = function(dir){
26381     var proxy = new Roo.Element(document.createElement("div"));
26382     proxy.unselectable();
26383     var cls = 'x-splitbar-proxy';
26384     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26385     document.body.appendChild(proxy.dom);
26386     return proxy.dom;
26387 };
26388
26389 /** 
26390  * @class Roo.SplitBar.BasicLayoutAdapter
26391  * Default Adapter. It assumes the splitter and resizing element are not positioned
26392  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26393  */
26394 Roo.SplitBar.BasicLayoutAdapter = function(){
26395 };
26396
26397 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26398     // do nothing for now
26399     init : function(s){
26400     
26401     },
26402     /**
26403      * Called before drag operations to get the current size of the resizing element. 
26404      * @param {Roo.SplitBar} s The SplitBar using this adapter
26405      */
26406      getElementSize : function(s){
26407         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26408             return s.resizingEl.getWidth();
26409         }else{
26410             return s.resizingEl.getHeight();
26411         }
26412     },
26413     
26414     /**
26415      * Called after drag operations to set the size of the resizing element.
26416      * @param {Roo.SplitBar} s The SplitBar using this adapter
26417      * @param {Number} newSize The new size to set
26418      * @param {Function} onComplete A function to be invoked when resizing is complete
26419      */
26420     setElementSize : function(s, newSize, onComplete){
26421         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26422             if(!s.animate){
26423                 s.resizingEl.setWidth(newSize);
26424                 if(onComplete){
26425                     onComplete(s, newSize);
26426                 }
26427             }else{
26428                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26429             }
26430         }else{
26431             
26432             if(!s.animate){
26433                 s.resizingEl.setHeight(newSize);
26434                 if(onComplete){
26435                     onComplete(s, newSize);
26436                 }
26437             }else{
26438                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26439             }
26440         }
26441     }
26442 };
26443
26444 /** 
26445  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26446  * @extends Roo.SplitBar.BasicLayoutAdapter
26447  * Adapter that  moves the splitter element to align with the resized sizing element. 
26448  * Used with an absolute positioned SplitBar.
26449  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26450  * document.body, make sure you assign an id to the body element.
26451  */
26452 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26453     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26454     this.container = Roo.get(container);
26455 };
26456
26457 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26458     init : function(s){
26459         this.basic.init(s);
26460     },
26461     
26462     getElementSize : function(s){
26463         return this.basic.getElementSize(s);
26464     },
26465     
26466     setElementSize : function(s, newSize, onComplete){
26467         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26468     },
26469     
26470     moveSplitter : function(s){
26471         var yes = Roo.SplitBar;
26472         switch(s.placement){
26473             case yes.LEFT:
26474                 s.el.setX(s.resizingEl.getRight());
26475                 break;
26476             case yes.RIGHT:
26477                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26478                 break;
26479             case yes.TOP:
26480                 s.el.setY(s.resizingEl.getBottom());
26481                 break;
26482             case yes.BOTTOM:
26483                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26484                 break;
26485         }
26486     }
26487 };
26488
26489 /**
26490  * Orientation constant - Create a vertical SplitBar
26491  * @static
26492  * @type Number
26493  */
26494 Roo.SplitBar.VERTICAL = 1;
26495
26496 /**
26497  * Orientation constant - Create a horizontal SplitBar
26498  * @static
26499  * @type Number
26500  */
26501 Roo.SplitBar.HORIZONTAL = 2;
26502
26503 /**
26504  * Placement constant - The resizing element is to the left of the splitter element
26505  * @static
26506  * @type Number
26507  */
26508 Roo.SplitBar.LEFT = 1;
26509
26510 /**
26511  * Placement constant - The resizing element is to the right of the splitter element
26512  * @static
26513  * @type Number
26514  */
26515 Roo.SplitBar.RIGHT = 2;
26516
26517 /**
26518  * Placement constant - The resizing element is positioned above the splitter element
26519  * @static
26520  * @type Number
26521  */
26522 Roo.SplitBar.TOP = 3;
26523
26524 /**
26525  * Placement constant - The resizing element is positioned under splitter element
26526  * @static
26527  * @type Number
26528  */
26529 Roo.SplitBar.BOTTOM = 4;
26530 /*
26531  * Based on:
26532  * Ext JS Library 1.1.1
26533  * Copyright(c) 2006-2007, Ext JS, LLC.
26534  *
26535  * Originally Released Under LGPL - original licence link has changed is not relivant.
26536  *
26537  * Fork - LGPL
26538  * <script type="text/javascript">
26539  */
26540
26541 /**
26542  * @class Roo.View
26543  * @extends Roo.util.Observable
26544  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26545  * This class also supports single and multi selection modes. <br>
26546  * Create a data model bound view:
26547  <pre><code>
26548  var store = new Roo.data.Store(...);
26549
26550  var view = new Roo.View({
26551     el : "my-element",
26552     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26553  
26554     singleSelect: true,
26555     selectedClass: "ydataview-selected",
26556     store: store
26557  });
26558
26559  // listen for node click?
26560  view.on("click", function(vw, index, node, e){
26561  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26562  });
26563
26564  // load XML data
26565  dataModel.load("foobar.xml");
26566  </code></pre>
26567  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26568  * <br><br>
26569  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26570  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26571  * 
26572  * Note: old style constructor is still suported (container, template, config)
26573  * 
26574  * @constructor
26575  * Create a new View
26576  * @param {Object} config The config object
26577  * 
26578  */
26579 Roo.View = function(config, depreciated_tpl, depreciated_config){
26580     
26581     this.parent = false;
26582     
26583     if (typeof(depreciated_tpl) == 'undefined') {
26584         // new way.. - universal constructor.
26585         Roo.apply(this, config);
26586         this.el  = Roo.get(this.el);
26587     } else {
26588         // old format..
26589         this.el  = Roo.get(config);
26590         this.tpl = depreciated_tpl;
26591         Roo.apply(this, depreciated_config);
26592     }
26593     this.wrapEl  = this.el.wrap().wrap();
26594     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26595     
26596     
26597     if(typeof(this.tpl) == "string"){
26598         this.tpl = new Roo.Template(this.tpl);
26599     } else {
26600         // support xtype ctors..
26601         this.tpl = new Roo.factory(this.tpl, Roo);
26602     }
26603     
26604     
26605     this.tpl.compile();
26606     
26607     /** @private */
26608     this.addEvents({
26609         /**
26610          * @event beforeclick
26611          * Fires before a click is processed. Returns false to cancel the default action.
26612          * @param {Roo.View} this
26613          * @param {Number} index The index of the target node
26614          * @param {HTMLElement} node The target node
26615          * @param {Roo.EventObject} e The raw event object
26616          */
26617             "beforeclick" : true,
26618         /**
26619          * @event click
26620          * Fires when a template node is clicked.
26621          * @param {Roo.View} this
26622          * @param {Number} index The index of the target node
26623          * @param {HTMLElement} node The target node
26624          * @param {Roo.EventObject} e The raw event object
26625          */
26626             "click" : true,
26627         /**
26628          * @event dblclick
26629          * Fires when a template node is double clicked.
26630          * @param {Roo.View} this
26631          * @param {Number} index The index of the target node
26632          * @param {HTMLElement} node The target node
26633          * @param {Roo.EventObject} e The raw event object
26634          */
26635             "dblclick" : true,
26636         /**
26637          * @event contextmenu
26638          * Fires when a template node is right clicked.
26639          * @param {Roo.View} this
26640          * @param {Number} index The index of the target node
26641          * @param {HTMLElement} node The target node
26642          * @param {Roo.EventObject} e The raw event object
26643          */
26644             "contextmenu" : true,
26645         /**
26646          * @event selectionchange
26647          * Fires when the selected nodes change.
26648          * @param {Roo.View} this
26649          * @param {Array} selections Array of the selected nodes
26650          */
26651             "selectionchange" : true,
26652     
26653         /**
26654          * @event beforeselect
26655          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26656          * @param {Roo.View} this
26657          * @param {HTMLElement} node The node to be selected
26658          * @param {Array} selections Array of currently selected nodes
26659          */
26660             "beforeselect" : true,
26661         /**
26662          * @event preparedata
26663          * Fires on every row to render, to allow you to change the data.
26664          * @param {Roo.View} this
26665          * @param {Object} data to be rendered (change this)
26666          */
26667           "preparedata" : true
26668           
26669           
26670         });
26671
26672
26673
26674     this.el.on({
26675         "click": this.onClick,
26676         "dblclick": this.onDblClick,
26677         "contextmenu": this.onContextMenu,
26678         scope:this
26679     });
26680
26681     this.selections = [];
26682     this.nodes = [];
26683     this.cmp = new Roo.CompositeElementLite([]);
26684     if(this.store){
26685         this.store = Roo.factory(this.store, Roo.data);
26686         this.setStore(this.store, true);
26687     }
26688     
26689     if ( this.footer && this.footer.xtype) {
26690            
26691          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26692         
26693         this.footer.dataSource = this.store;
26694         this.footer.container = fctr;
26695         this.footer = Roo.factory(this.footer, Roo);
26696         fctr.insertFirst(this.el);
26697         
26698         // this is a bit insane - as the paging toolbar seems to detach the el..
26699 //        dom.parentNode.parentNode.parentNode
26700          // they get detached?
26701     }
26702     
26703     
26704     Roo.View.superclass.constructor.call(this);
26705     
26706     
26707 };
26708
26709 Roo.extend(Roo.View, Roo.util.Observable, {
26710     
26711      /**
26712      * @cfg {Roo.data.Store} store Data store to load data from.
26713      */
26714     store : false,
26715     
26716     /**
26717      * @cfg {String|Roo.Element} el The container element.
26718      */
26719     el : '',
26720     
26721     /**
26722      * @cfg {String|Roo.Template} tpl The template used by this View 
26723      */
26724     tpl : false,
26725     /**
26726      * @cfg {String} dataName the named area of the template to use as the data area
26727      *                          Works with domtemplates roo-name="name"
26728      */
26729     dataName: false,
26730     /**
26731      * @cfg {String} selectedClass The css class to add to selected nodes
26732      */
26733     selectedClass : "x-view-selected",
26734      /**
26735      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26736      */
26737     emptyText : "",
26738     
26739     /**
26740      * @cfg {String} text to display on mask (default Loading)
26741      */
26742     mask : false,
26743     /**
26744      * @cfg {Boolean} multiSelect Allow multiple selection
26745      */
26746     multiSelect : false,
26747     /**
26748      * @cfg {Boolean} singleSelect Allow single selection
26749      */
26750     singleSelect:  false,
26751     
26752     /**
26753      * @cfg {Boolean} toggleSelect - selecting 
26754      */
26755     toggleSelect : false,
26756     
26757     /**
26758      * @cfg {Boolean} tickable - selecting 
26759      */
26760     tickable : false,
26761     
26762     /**
26763      * Returns the element this view is bound to.
26764      * @return {Roo.Element}
26765      */
26766     getEl : function(){
26767         return this.wrapEl;
26768     },
26769     
26770     
26771
26772     /**
26773      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26774      */
26775     refresh : function(){
26776         //Roo.log('refresh');
26777         var t = this.tpl;
26778         
26779         // if we are using something like 'domtemplate', then
26780         // the what gets used is:
26781         // t.applySubtemplate(NAME, data, wrapping data..)
26782         // the outer template then get' applied with
26783         //     the store 'extra data'
26784         // and the body get's added to the
26785         //      roo-name="data" node?
26786         //      <span class='roo-tpl-{name}'></span> ?????
26787         
26788         
26789         
26790         this.clearSelections();
26791         this.el.update("");
26792         var html = [];
26793         var records = this.store.getRange();
26794         if(records.length < 1) {
26795             
26796             // is this valid??  = should it render a template??
26797             
26798             this.el.update(this.emptyText);
26799             return;
26800         }
26801         var el = this.el;
26802         if (this.dataName) {
26803             this.el.update(t.apply(this.store.meta)); //????
26804             el = this.el.child('.roo-tpl-' + this.dataName);
26805         }
26806         
26807         for(var i = 0, len = records.length; i < len; i++){
26808             var data = this.prepareData(records[i].data, i, records[i]);
26809             this.fireEvent("preparedata", this, data, i, records[i]);
26810             
26811             var d = Roo.apply({}, data);
26812             
26813             if(this.tickable){
26814                 Roo.apply(d, {'roo-id' : Roo.id()});
26815                 
26816                 var _this = this;
26817             
26818                 Roo.each(this.parent.item, function(item){
26819                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26820                         return;
26821                     }
26822                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26823                 });
26824             }
26825             
26826             html[html.length] = Roo.util.Format.trim(
26827                 this.dataName ?
26828                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26829                     t.apply(d)
26830             );
26831         }
26832         
26833         
26834         
26835         el.update(html.join(""));
26836         this.nodes = el.dom.childNodes;
26837         this.updateIndexes(0);
26838     },
26839     
26840
26841     /**
26842      * Function to override to reformat the data that is sent to
26843      * the template for each node.
26844      * DEPRICATED - use the preparedata event handler.
26845      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26846      * a JSON object for an UpdateManager bound view).
26847      */
26848     prepareData : function(data, index, record)
26849     {
26850         this.fireEvent("preparedata", this, data, index, record);
26851         return data;
26852     },
26853
26854     onUpdate : function(ds, record){
26855         // Roo.log('on update');   
26856         this.clearSelections();
26857         var index = this.store.indexOf(record);
26858         var n = this.nodes[index];
26859         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26860         n.parentNode.removeChild(n);
26861         this.updateIndexes(index, index);
26862     },
26863
26864     
26865     
26866 // --------- FIXME     
26867     onAdd : function(ds, records, index)
26868     {
26869         //Roo.log(['on Add', ds, records, index] );        
26870         this.clearSelections();
26871         if(this.nodes.length == 0){
26872             this.refresh();
26873             return;
26874         }
26875         var n = this.nodes[index];
26876         for(var i = 0, len = records.length; i < len; i++){
26877             var d = this.prepareData(records[i].data, i, records[i]);
26878             if(n){
26879                 this.tpl.insertBefore(n, d);
26880             }else{
26881                 
26882                 this.tpl.append(this.el, d);
26883             }
26884         }
26885         this.updateIndexes(index);
26886     },
26887
26888     onRemove : function(ds, record, index){
26889        // Roo.log('onRemove');
26890         this.clearSelections();
26891         var el = this.dataName  ?
26892             this.el.child('.roo-tpl-' + this.dataName) :
26893             this.el; 
26894         
26895         el.dom.removeChild(this.nodes[index]);
26896         this.updateIndexes(index);
26897     },
26898
26899     /**
26900      * Refresh an individual node.
26901      * @param {Number} index
26902      */
26903     refreshNode : function(index){
26904         this.onUpdate(this.store, this.store.getAt(index));
26905     },
26906
26907     updateIndexes : function(startIndex, endIndex){
26908         var ns = this.nodes;
26909         startIndex = startIndex || 0;
26910         endIndex = endIndex || ns.length - 1;
26911         for(var i = startIndex; i <= endIndex; i++){
26912             ns[i].nodeIndex = i;
26913         }
26914     },
26915
26916     /**
26917      * Changes the data store this view uses and refresh the view.
26918      * @param {Store} store
26919      */
26920     setStore : function(store, initial){
26921         if(!initial && this.store){
26922             this.store.un("datachanged", this.refresh);
26923             this.store.un("add", this.onAdd);
26924             this.store.un("remove", this.onRemove);
26925             this.store.un("update", this.onUpdate);
26926             this.store.un("clear", this.refresh);
26927             this.store.un("beforeload", this.onBeforeLoad);
26928             this.store.un("load", this.onLoad);
26929             this.store.un("loadexception", this.onLoad);
26930         }
26931         if(store){
26932           
26933             store.on("datachanged", this.refresh, this);
26934             store.on("add", this.onAdd, this);
26935             store.on("remove", this.onRemove, this);
26936             store.on("update", this.onUpdate, this);
26937             store.on("clear", this.refresh, this);
26938             store.on("beforeload", this.onBeforeLoad, this);
26939             store.on("load", this.onLoad, this);
26940             store.on("loadexception", this.onLoad, this);
26941         }
26942         
26943         if(store){
26944             this.refresh();
26945         }
26946     },
26947     /**
26948      * onbeforeLoad - masks the loading area.
26949      *
26950      */
26951     onBeforeLoad : function(store,opts)
26952     {
26953          //Roo.log('onBeforeLoad');   
26954         if (!opts.add) {
26955             this.el.update("");
26956         }
26957         this.el.mask(this.mask ? this.mask : "Loading" ); 
26958     },
26959     onLoad : function ()
26960     {
26961         this.el.unmask();
26962     },
26963     
26964
26965     /**
26966      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26967      * @param {HTMLElement} node
26968      * @return {HTMLElement} The template node
26969      */
26970     findItemFromChild : function(node){
26971         var el = this.dataName  ?
26972             this.el.child('.roo-tpl-' + this.dataName,true) :
26973             this.el.dom; 
26974         
26975         if(!node || node.parentNode == el){
26976                     return node;
26977             }
26978             var p = node.parentNode;
26979             while(p && p != el){
26980             if(p.parentNode == el){
26981                 return p;
26982             }
26983             p = p.parentNode;
26984         }
26985             return null;
26986     },
26987
26988     /** @ignore */
26989     onClick : function(e){
26990         var item = this.findItemFromChild(e.getTarget());
26991         if(item){
26992             var index = this.indexOf(item);
26993             if(this.onItemClick(item, index, e) !== false){
26994                 this.fireEvent("click", this, index, item, e);
26995             }
26996         }else{
26997             this.clearSelections();
26998         }
26999     },
27000
27001     /** @ignore */
27002     onContextMenu : function(e){
27003         var item = this.findItemFromChild(e.getTarget());
27004         if(item){
27005             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
27006         }
27007     },
27008
27009     /** @ignore */
27010     onDblClick : function(e){
27011         var item = this.findItemFromChild(e.getTarget());
27012         if(item){
27013             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27014         }
27015     },
27016
27017     onItemClick : function(item, index, e)
27018     {
27019         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27020             return false;
27021         }
27022         if (this.toggleSelect) {
27023             var m = this.isSelected(item) ? 'unselect' : 'select';
27024             //Roo.log(m);
27025             var _t = this;
27026             _t[m](item, true, false);
27027             return true;
27028         }
27029         if(this.multiSelect || this.singleSelect){
27030             if(this.multiSelect && e.shiftKey && this.lastSelection){
27031                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27032             }else{
27033                 this.select(item, this.multiSelect && e.ctrlKey);
27034                 this.lastSelection = item;
27035             }
27036             
27037             if(!this.tickable){
27038                 e.preventDefault();
27039             }
27040             
27041         }
27042         return true;
27043     },
27044
27045     /**
27046      * Get the number of selected nodes.
27047      * @return {Number}
27048      */
27049     getSelectionCount : function(){
27050         return this.selections.length;
27051     },
27052
27053     /**
27054      * Get the currently selected nodes.
27055      * @return {Array} An array of HTMLElements
27056      */
27057     getSelectedNodes : function(){
27058         return this.selections;
27059     },
27060
27061     /**
27062      * Get the indexes of the selected nodes.
27063      * @return {Array}
27064      */
27065     getSelectedIndexes : function(){
27066         var indexes = [], s = this.selections;
27067         for(var i = 0, len = s.length; i < len; i++){
27068             indexes.push(s[i].nodeIndex);
27069         }
27070         return indexes;
27071     },
27072
27073     /**
27074      * Clear all selections
27075      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27076      */
27077     clearSelections : function(suppressEvent){
27078         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27079             this.cmp.elements = this.selections;
27080             this.cmp.removeClass(this.selectedClass);
27081             this.selections = [];
27082             if(!suppressEvent){
27083                 this.fireEvent("selectionchange", this, this.selections);
27084             }
27085         }
27086     },
27087
27088     /**
27089      * Returns true if the passed node is selected
27090      * @param {HTMLElement/Number} node The node or node index
27091      * @return {Boolean}
27092      */
27093     isSelected : function(node){
27094         var s = this.selections;
27095         if(s.length < 1){
27096             return false;
27097         }
27098         node = this.getNode(node);
27099         return s.indexOf(node) !== -1;
27100     },
27101
27102     /**
27103      * Selects nodes.
27104      * @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
27105      * @param {Boolean} keepExisting (optional) true to keep existing selections
27106      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27107      */
27108     select : function(nodeInfo, keepExisting, suppressEvent){
27109         if(nodeInfo instanceof Array){
27110             if(!keepExisting){
27111                 this.clearSelections(true);
27112             }
27113             for(var i = 0, len = nodeInfo.length; i < len; i++){
27114                 this.select(nodeInfo[i], true, true);
27115             }
27116             return;
27117         } 
27118         var node = this.getNode(nodeInfo);
27119         if(!node || this.isSelected(node)){
27120             return; // already selected.
27121         }
27122         if(!keepExisting){
27123             this.clearSelections(true);
27124         }
27125         
27126         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27127             Roo.fly(node).addClass(this.selectedClass);
27128             this.selections.push(node);
27129             if(!suppressEvent){
27130                 this.fireEvent("selectionchange", this, this.selections);
27131             }
27132         }
27133         
27134         
27135     },
27136       /**
27137      * Unselects nodes.
27138      * @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
27139      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27140      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27141      */
27142     unselect : function(nodeInfo, keepExisting, suppressEvent)
27143     {
27144         if(nodeInfo instanceof Array){
27145             Roo.each(this.selections, function(s) {
27146                 this.unselect(s, nodeInfo);
27147             }, this);
27148             return;
27149         }
27150         var node = this.getNode(nodeInfo);
27151         if(!node || !this.isSelected(node)){
27152             //Roo.log("not selected");
27153             return; // not selected.
27154         }
27155         // fireevent???
27156         var ns = [];
27157         Roo.each(this.selections, function(s) {
27158             if (s == node ) {
27159                 Roo.fly(node).removeClass(this.selectedClass);
27160
27161                 return;
27162             }
27163             ns.push(s);
27164         },this);
27165         
27166         this.selections= ns;
27167         this.fireEvent("selectionchange", this, this.selections);
27168     },
27169
27170     /**
27171      * Gets a template node.
27172      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27173      * @return {HTMLElement} The node or null if it wasn't found
27174      */
27175     getNode : function(nodeInfo){
27176         if(typeof nodeInfo == "string"){
27177             return document.getElementById(nodeInfo);
27178         }else if(typeof nodeInfo == "number"){
27179             return this.nodes[nodeInfo];
27180         }
27181         return nodeInfo;
27182     },
27183
27184     /**
27185      * Gets a range template nodes.
27186      * @param {Number} startIndex
27187      * @param {Number} endIndex
27188      * @return {Array} An array of nodes
27189      */
27190     getNodes : function(start, end){
27191         var ns = this.nodes;
27192         start = start || 0;
27193         end = typeof end == "undefined" ? ns.length - 1 : end;
27194         var nodes = [];
27195         if(start <= end){
27196             for(var i = start; i <= end; i++){
27197                 nodes.push(ns[i]);
27198             }
27199         } else{
27200             for(var i = start; i >= end; i--){
27201                 nodes.push(ns[i]);
27202             }
27203         }
27204         return nodes;
27205     },
27206
27207     /**
27208      * Finds the index of the passed node
27209      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27210      * @return {Number} The index of the node or -1
27211      */
27212     indexOf : function(node){
27213         node = this.getNode(node);
27214         if(typeof node.nodeIndex == "number"){
27215             return node.nodeIndex;
27216         }
27217         var ns = this.nodes;
27218         for(var i = 0, len = ns.length; i < len; i++){
27219             if(ns[i] == node){
27220                 return i;
27221             }
27222         }
27223         return -1;
27224     }
27225 });
27226 /*
27227  * Based on:
27228  * Ext JS Library 1.1.1
27229  * Copyright(c) 2006-2007, Ext JS, LLC.
27230  *
27231  * Originally Released Under LGPL - original licence link has changed is not relivant.
27232  *
27233  * Fork - LGPL
27234  * <script type="text/javascript">
27235  */
27236
27237 /**
27238  * @class Roo.JsonView
27239  * @extends Roo.View
27240  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27241 <pre><code>
27242 var view = new Roo.JsonView({
27243     container: "my-element",
27244     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27245     multiSelect: true, 
27246     jsonRoot: "data" 
27247 });
27248
27249 // listen for node click?
27250 view.on("click", function(vw, index, node, e){
27251     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27252 });
27253
27254 // direct load of JSON data
27255 view.load("foobar.php");
27256
27257 // Example from my blog list
27258 var tpl = new Roo.Template(
27259     '&lt;div class="entry"&gt;' +
27260     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27261     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27262     "&lt;/div&gt;&lt;hr /&gt;"
27263 );
27264
27265 var moreView = new Roo.JsonView({
27266     container :  "entry-list", 
27267     template : tpl,
27268     jsonRoot: "posts"
27269 });
27270 moreView.on("beforerender", this.sortEntries, this);
27271 moreView.load({
27272     url: "/blog/get-posts.php",
27273     params: "allposts=true",
27274     text: "Loading Blog Entries..."
27275 });
27276 </code></pre>
27277
27278 * Note: old code is supported with arguments : (container, template, config)
27279
27280
27281  * @constructor
27282  * Create a new JsonView
27283  * 
27284  * @param {Object} config The config object
27285  * 
27286  */
27287 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27288     
27289     
27290     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27291
27292     var um = this.el.getUpdateManager();
27293     um.setRenderer(this);
27294     um.on("update", this.onLoad, this);
27295     um.on("failure", this.onLoadException, this);
27296
27297     /**
27298      * @event beforerender
27299      * Fires before rendering of the downloaded JSON data.
27300      * @param {Roo.JsonView} this
27301      * @param {Object} data The JSON data loaded
27302      */
27303     /**
27304      * @event load
27305      * Fires when data is loaded.
27306      * @param {Roo.JsonView} this
27307      * @param {Object} data The JSON data loaded
27308      * @param {Object} response The raw Connect response object
27309      */
27310     /**
27311      * @event loadexception
27312      * Fires when loading fails.
27313      * @param {Roo.JsonView} this
27314      * @param {Object} response The raw Connect response object
27315      */
27316     this.addEvents({
27317         'beforerender' : true,
27318         'load' : true,
27319         'loadexception' : true
27320     });
27321 };
27322 Roo.extend(Roo.JsonView, Roo.View, {
27323     /**
27324      * @type {String} The root property in the loaded JSON object that contains the data
27325      */
27326     jsonRoot : "",
27327
27328     /**
27329      * Refreshes the view.
27330      */
27331     refresh : function(){
27332         this.clearSelections();
27333         this.el.update("");
27334         var html = [];
27335         var o = this.jsonData;
27336         if(o && o.length > 0){
27337             for(var i = 0, len = o.length; i < len; i++){
27338                 var data = this.prepareData(o[i], i, o);
27339                 html[html.length] = this.tpl.apply(data);
27340             }
27341         }else{
27342             html.push(this.emptyText);
27343         }
27344         this.el.update(html.join(""));
27345         this.nodes = this.el.dom.childNodes;
27346         this.updateIndexes(0);
27347     },
27348
27349     /**
27350      * 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.
27351      * @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:
27352      <pre><code>
27353      view.load({
27354          url: "your-url.php",
27355          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27356          callback: yourFunction,
27357          scope: yourObject, //(optional scope)
27358          discardUrl: false,
27359          nocache: false,
27360          text: "Loading...",
27361          timeout: 30,
27362          scripts: false
27363      });
27364      </code></pre>
27365      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27366      * 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.
27367      * @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}
27368      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27369      * @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.
27370      */
27371     load : function(){
27372         var um = this.el.getUpdateManager();
27373         um.update.apply(um, arguments);
27374     },
27375
27376     // note - render is a standard framework call...
27377     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27378     render : function(el, response){
27379         
27380         this.clearSelections();
27381         this.el.update("");
27382         var o;
27383         try{
27384             if (response != '') {
27385                 o = Roo.util.JSON.decode(response.responseText);
27386                 if(this.jsonRoot){
27387                     
27388                     o = o[this.jsonRoot];
27389                 }
27390             }
27391         } catch(e){
27392         }
27393         /**
27394          * The current JSON data or null
27395          */
27396         this.jsonData = o;
27397         this.beforeRender();
27398         this.refresh();
27399     },
27400
27401 /**
27402  * Get the number of records in the current JSON dataset
27403  * @return {Number}
27404  */
27405     getCount : function(){
27406         return this.jsonData ? this.jsonData.length : 0;
27407     },
27408
27409 /**
27410  * Returns the JSON object for the specified node(s)
27411  * @param {HTMLElement/Array} node The node or an array of nodes
27412  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27413  * you get the JSON object for the node
27414  */
27415     getNodeData : function(node){
27416         if(node instanceof Array){
27417             var data = [];
27418             for(var i = 0, len = node.length; i < len; i++){
27419                 data.push(this.getNodeData(node[i]));
27420             }
27421             return data;
27422         }
27423         return this.jsonData[this.indexOf(node)] || null;
27424     },
27425
27426     beforeRender : function(){
27427         this.snapshot = this.jsonData;
27428         if(this.sortInfo){
27429             this.sort.apply(this, this.sortInfo);
27430         }
27431         this.fireEvent("beforerender", this, this.jsonData);
27432     },
27433
27434     onLoad : function(el, o){
27435         this.fireEvent("load", this, this.jsonData, o);
27436     },
27437
27438     onLoadException : function(el, o){
27439         this.fireEvent("loadexception", this, o);
27440     },
27441
27442 /**
27443  * Filter the data by a specific property.
27444  * @param {String} property A property on your JSON objects
27445  * @param {String/RegExp} value Either string that the property values
27446  * should start with, or a RegExp to test against the property
27447  */
27448     filter : function(property, value){
27449         if(this.jsonData){
27450             var data = [];
27451             var ss = this.snapshot;
27452             if(typeof value == "string"){
27453                 var vlen = value.length;
27454                 if(vlen == 0){
27455                     this.clearFilter();
27456                     return;
27457                 }
27458                 value = value.toLowerCase();
27459                 for(var i = 0, len = ss.length; i < len; i++){
27460                     var o = ss[i];
27461                     if(o[property].substr(0, vlen).toLowerCase() == value){
27462                         data.push(o);
27463                     }
27464                 }
27465             } else if(value.exec){ // regex?
27466                 for(var i = 0, len = ss.length; i < len; i++){
27467                     var o = ss[i];
27468                     if(value.test(o[property])){
27469                         data.push(o);
27470                     }
27471                 }
27472             } else{
27473                 return;
27474             }
27475             this.jsonData = data;
27476             this.refresh();
27477         }
27478     },
27479
27480 /**
27481  * Filter by a function. The passed function will be called with each
27482  * object in the current dataset. If the function returns true the value is kept,
27483  * otherwise it is filtered.
27484  * @param {Function} fn
27485  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27486  */
27487     filterBy : function(fn, scope){
27488         if(this.jsonData){
27489             var data = [];
27490             var ss = this.snapshot;
27491             for(var i = 0, len = ss.length; i < len; i++){
27492                 var o = ss[i];
27493                 if(fn.call(scope || this, o)){
27494                     data.push(o);
27495                 }
27496             }
27497             this.jsonData = data;
27498             this.refresh();
27499         }
27500     },
27501
27502 /**
27503  * Clears the current filter.
27504  */
27505     clearFilter : function(){
27506         if(this.snapshot && this.jsonData != this.snapshot){
27507             this.jsonData = this.snapshot;
27508             this.refresh();
27509         }
27510     },
27511
27512
27513 /**
27514  * Sorts the data for this view and refreshes it.
27515  * @param {String} property A property on your JSON objects to sort on
27516  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27517  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27518  */
27519     sort : function(property, dir, sortType){
27520         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27521         if(this.jsonData){
27522             var p = property;
27523             var dsc = dir && dir.toLowerCase() == "desc";
27524             var f = function(o1, o2){
27525                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27526                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27527                 ;
27528                 if(v1 < v2){
27529                     return dsc ? +1 : -1;
27530                 } else if(v1 > v2){
27531                     return dsc ? -1 : +1;
27532                 } else{
27533                     return 0;
27534                 }
27535             };
27536             this.jsonData.sort(f);
27537             this.refresh();
27538             if(this.jsonData != this.snapshot){
27539                 this.snapshot.sort(f);
27540             }
27541         }
27542     }
27543 });/*
27544  * Based on:
27545  * Ext JS Library 1.1.1
27546  * Copyright(c) 2006-2007, Ext JS, LLC.
27547  *
27548  * Originally Released Under LGPL - original licence link has changed is not relivant.
27549  *
27550  * Fork - LGPL
27551  * <script type="text/javascript">
27552  */
27553  
27554
27555 /**
27556  * @class Roo.ColorPalette
27557  * @extends Roo.Component
27558  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27559  * Here's an example of typical usage:
27560  * <pre><code>
27561 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27562 cp.render('my-div');
27563
27564 cp.on('select', function(palette, selColor){
27565     // do something with selColor
27566 });
27567 </code></pre>
27568  * @constructor
27569  * Create a new ColorPalette
27570  * @param {Object} config The config object
27571  */
27572 Roo.ColorPalette = function(config){
27573     Roo.ColorPalette.superclass.constructor.call(this, config);
27574     this.addEvents({
27575         /**
27576              * @event select
27577              * Fires when a color is selected
27578              * @param {ColorPalette} this
27579              * @param {String} color The 6-digit color hex code (without the # symbol)
27580              */
27581         select: true
27582     });
27583
27584     if(this.handler){
27585         this.on("select", this.handler, this.scope, true);
27586     }
27587 };
27588 Roo.extend(Roo.ColorPalette, Roo.Component, {
27589     /**
27590      * @cfg {String} itemCls
27591      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27592      */
27593     itemCls : "x-color-palette",
27594     /**
27595      * @cfg {String} value
27596      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27597      * the hex codes are case-sensitive.
27598      */
27599     value : null,
27600     clickEvent:'click',
27601     // private
27602     ctype: "Roo.ColorPalette",
27603
27604     /**
27605      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27606      */
27607     allowReselect : false,
27608
27609     /**
27610      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27611      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27612      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27613      * of colors with the width setting until the box is symmetrical.</p>
27614      * <p>You can override individual colors if needed:</p>
27615      * <pre><code>
27616 var cp = new Roo.ColorPalette();
27617 cp.colors[0] = "FF0000";  // change the first box to red
27618 </code></pre>
27619
27620 Or you can provide a custom array of your own for complete control:
27621 <pre><code>
27622 var cp = new Roo.ColorPalette();
27623 cp.colors = ["000000", "993300", "333300"];
27624 </code></pre>
27625      * @type Array
27626      */
27627     colors : [
27628         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27629         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27630         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27631         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27632         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27633     ],
27634
27635     // private
27636     onRender : function(container, position){
27637         var t = new Roo.MasterTemplate(
27638             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27639         );
27640         var c = this.colors;
27641         for(var i = 0, len = c.length; i < len; i++){
27642             t.add([c[i]]);
27643         }
27644         var el = document.createElement("div");
27645         el.className = this.itemCls;
27646         t.overwrite(el);
27647         container.dom.insertBefore(el, position);
27648         this.el = Roo.get(el);
27649         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27650         if(this.clickEvent != 'click'){
27651             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27652         }
27653     },
27654
27655     // private
27656     afterRender : function(){
27657         Roo.ColorPalette.superclass.afterRender.call(this);
27658         if(this.value){
27659             var s = this.value;
27660             this.value = null;
27661             this.select(s);
27662         }
27663     },
27664
27665     // private
27666     handleClick : function(e, t){
27667         e.preventDefault();
27668         if(!this.disabled){
27669             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27670             this.select(c.toUpperCase());
27671         }
27672     },
27673
27674     /**
27675      * Selects the specified color in the palette (fires the select event)
27676      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27677      */
27678     select : function(color){
27679         color = color.replace("#", "");
27680         if(color != this.value || this.allowReselect){
27681             var el = this.el;
27682             if(this.value){
27683                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27684             }
27685             el.child("a.color-"+color).addClass("x-color-palette-sel");
27686             this.value = color;
27687             this.fireEvent("select", this, color);
27688         }
27689     }
27690 });/*
27691  * Based on:
27692  * Ext JS Library 1.1.1
27693  * Copyright(c) 2006-2007, Ext JS, LLC.
27694  *
27695  * Originally Released Under LGPL - original licence link has changed is not relivant.
27696  *
27697  * Fork - LGPL
27698  * <script type="text/javascript">
27699  */
27700  
27701 /**
27702  * @class Roo.DatePicker
27703  * @extends Roo.Component
27704  * Simple date picker class.
27705  * @constructor
27706  * Create a new DatePicker
27707  * @param {Object} config The config object
27708  */
27709 Roo.DatePicker = function(config){
27710     Roo.DatePicker.superclass.constructor.call(this, config);
27711
27712     this.value = config && config.value ?
27713                  config.value.clearTime() : new Date().clearTime();
27714
27715     this.addEvents({
27716         /**
27717              * @event select
27718              * Fires when a date is selected
27719              * @param {DatePicker} this
27720              * @param {Date} date The selected date
27721              */
27722         'select': true,
27723         /**
27724              * @event monthchange
27725              * Fires when the displayed month changes 
27726              * @param {DatePicker} this
27727              * @param {Date} date The selected month
27728              */
27729         'monthchange': true
27730     });
27731
27732     if(this.handler){
27733         this.on("select", this.handler,  this.scope || this);
27734     }
27735     // build the disabledDatesRE
27736     if(!this.disabledDatesRE && this.disabledDates){
27737         var dd = this.disabledDates;
27738         var re = "(?:";
27739         for(var i = 0; i < dd.length; i++){
27740             re += dd[i];
27741             if(i != dd.length-1) {
27742                 re += "|";
27743             }
27744         }
27745         this.disabledDatesRE = new RegExp(re + ")");
27746     }
27747 };
27748
27749 Roo.extend(Roo.DatePicker, Roo.Component, {
27750     /**
27751      * @cfg {String} todayText
27752      * The text to display on the button that selects the current date (defaults to "Today")
27753      */
27754     todayText : "Today",
27755     /**
27756      * @cfg {String} okText
27757      * The text to display on the ok button
27758      */
27759     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27760     /**
27761      * @cfg {String} cancelText
27762      * The text to display on the cancel button
27763      */
27764     cancelText : "Cancel",
27765     /**
27766      * @cfg {String} todayTip
27767      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27768      */
27769     todayTip : "{0} (Spacebar)",
27770     /**
27771      * @cfg {Date} minDate
27772      * Minimum allowable date (JavaScript date object, defaults to null)
27773      */
27774     minDate : null,
27775     /**
27776      * @cfg {Date} maxDate
27777      * Maximum allowable date (JavaScript date object, defaults to null)
27778      */
27779     maxDate : null,
27780     /**
27781      * @cfg {String} minText
27782      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27783      */
27784     minText : "This date is before the minimum date",
27785     /**
27786      * @cfg {String} maxText
27787      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27788      */
27789     maxText : "This date is after the maximum date",
27790     /**
27791      * @cfg {String} format
27792      * The default date format string which can be overriden for localization support.  The format must be
27793      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27794      */
27795     format : "m/d/y",
27796     /**
27797      * @cfg {Array} disabledDays
27798      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27799      */
27800     disabledDays : null,
27801     /**
27802      * @cfg {String} disabledDaysText
27803      * The tooltip to display when the date falls on a disabled day (defaults to "")
27804      */
27805     disabledDaysText : "",
27806     /**
27807      * @cfg {RegExp} disabledDatesRE
27808      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27809      */
27810     disabledDatesRE : null,
27811     /**
27812      * @cfg {String} disabledDatesText
27813      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27814      */
27815     disabledDatesText : "",
27816     /**
27817      * @cfg {Boolean} constrainToViewport
27818      * True to constrain the date picker to the viewport (defaults to true)
27819      */
27820     constrainToViewport : true,
27821     /**
27822      * @cfg {Array} monthNames
27823      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27824      */
27825     monthNames : Date.monthNames,
27826     /**
27827      * @cfg {Array} dayNames
27828      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27829      */
27830     dayNames : Date.dayNames,
27831     /**
27832      * @cfg {String} nextText
27833      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27834      */
27835     nextText: 'Next Month (Control+Right)',
27836     /**
27837      * @cfg {String} prevText
27838      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27839      */
27840     prevText: 'Previous Month (Control+Left)',
27841     /**
27842      * @cfg {String} monthYearText
27843      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27844      */
27845     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27846     /**
27847      * @cfg {Number} startDay
27848      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27849      */
27850     startDay : 0,
27851     /**
27852      * @cfg {Bool} showClear
27853      * Show a clear button (usefull for date form elements that can be blank.)
27854      */
27855     
27856     showClear: false,
27857     
27858     /**
27859      * Sets the value of the date field
27860      * @param {Date} value The date to set
27861      */
27862     setValue : function(value){
27863         var old = this.value;
27864         
27865         if (typeof(value) == 'string') {
27866          
27867             value = Date.parseDate(value, this.format);
27868         }
27869         if (!value) {
27870             value = new Date();
27871         }
27872         
27873         this.value = value.clearTime(true);
27874         if(this.el){
27875             this.update(this.value);
27876         }
27877     },
27878
27879     /**
27880      * Gets the current selected value of the date field
27881      * @return {Date} The selected date
27882      */
27883     getValue : function(){
27884         return this.value;
27885     },
27886
27887     // private
27888     focus : function(){
27889         if(this.el){
27890             this.update(this.activeDate);
27891         }
27892     },
27893
27894     // privateval
27895     onRender : function(container, position){
27896         
27897         var m = [
27898              '<table cellspacing="0">',
27899                 '<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>',
27900                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27901         var dn = this.dayNames;
27902         for(var i = 0; i < 7; i++){
27903             var d = this.startDay+i;
27904             if(d > 6){
27905                 d = d-7;
27906             }
27907             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27908         }
27909         m[m.length] = "</tr></thead><tbody><tr>";
27910         for(var i = 0; i < 42; i++) {
27911             if(i % 7 == 0 && i != 0){
27912                 m[m.length] = "</tr><tr>";
27913             }
27914             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27915         }
27916         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27917             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27918
27919         var el = document.createElement("div");
27920         el.className = "x-date-picker";
27921         el.innerHTML = m.join("");
27922
27923         container.dom.insertBefore(el, position);
27924
27925         this.el = Roo.get(el);
27926         this.eventEl = Roo.get(el.firstChild);
27927
27928         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27929             handler: this.showPrevMonth,
27930             scope: this,
27931             preventDefault:true,
27932             stopDefault:true
27933         });
27934
27935         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27936             handler: this.showNextMonth,
27937             scope: this,
27938             preventDefault:true,
27939             stopDefault:true
27940         });
27941
27942         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27943
27944         this.monthPicker = this.el.down('div.x-date-mp');
27945         this.monthPicker.enableDisplayMode('block');
27946         
27947         var kn = new Roo.KeyNav(this.eventEl, {
27948             "left" : function(e){
27949                 e.ctrlKey ?
27950                     this.showPrevMonth() :
27951                     this.update(this.activeDate.add("d", -1));
27952             },
27953
27954             "right" : function(e){
27955                 e.ctrlKey ?
27956                     this.showNextMonth() :
27957                     this.update(this.activeDate.add("d", 1));
27958             },
27959
27960             "up" : function(e){
27961                 e.ctrlKey ?
27962                     this.showNextYear() :
27963                     this.update(this.activeDate.add("d", -7));
27964             },
27965
27966             "down" : function(e){
27967                 e.ctrlKey ?
27968                     this.showPrevYear() :
27969                     this.update(this.activeDate.add("d", 7));
27970             },
27971
27972             "pageUp" : function(e){
27973                 this.showNextMonth();
27974             },
27975
27976             "pageDown" : function(e){
27977                 this.showPrevMonth();
27978             },
27979
27980             "enter" : function(e){
27981                 e.stopPropagation();
27982                 return true;
27983             },
27984
27985             scope : this
27986         });
27987
27988         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27989
27990         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27991
27992         this.el.unselectable();
27993         
27994         this.cells = this.el.select("table.x-date-inner tbody td");
27995         this.textNodes = this.el.query("table.x-date-inner tbody span");
27996
27997         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27998             text: "&#160;",
27999             tooltip: this.monthYearText
28000         });
28001
28002         this.mbtn.on('click', this.showMonthPicker, this);
28003         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
28004
28005
28006         var today = (new Date()).dateFormat(this.format);
28007         
28008         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
28009         if (this.showClear) {
28010             baseTb.add( new Roo.Toolbar.Fill());
28011         }
28012         baseTb.add({
28013             text: String.format(this.todayText, today),
28014             tooltip: String.format(this.todayTip, today),
28015             handler: this.selectToday,
28016             scope: this
28017         });
28018         
28019         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28020             
28021         //});
28022         if (this.showClear) {
28023             
28024             baseTb.add( new Roo.Toolbar.Fill());
28025             baseTb.add({
28026                 text: '&#160;',
28027                 cls: 'x-btn-icon x-btn-clear',
28028                 handler: function() {
28029                     //this.value = '';
28030                     this.fireEvent("select", this, '');
28031                 },
28032                 scope: this
28033             });
28034         }
28035         
28036         
28037         if(Roo.isIE){
28038             this.el.repaint();
28039         }
28040         this.update(this.value);
28041     },
28042
28043     createMonthPicker : function(){
28044         if(!this.monthPicker.dom.firstChild){
28045             var buf = ['<table border="0" cellspacing="0">'];
28046             for(var i = 0; i < 6; i++){
28047                 buf.push(
28048                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28049                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28050                     i == 0 ?
28051                     '<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>' :
28052                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28053                 );
28054             }
28055             buf.push(
28056                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28057                     this.okText,
28058                     '</button><button type="button" class="x-date-mp-cancel">',
28059                     this.cancelText,
28060                     '</button></td></tr>',
28061                 '</table>'
28062             );
28063             this.monthPicker.update(buf.join(''));
28064             this.monthPicker.on('click', this.onMonthClick, this);
28065             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28066
28067             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28068             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28069
28070             this.mpMonths.each(function(m, a, i){
28071                 i += 1;
28072                 if((i%2) == 0){
28073                     m.dom.xmonth = 5 + Math.round(i * .5);
28074                 }else{
28075                     m.dom.xmonth = Math.round((i-1) * .5);
28076                 }
28077             });
28078         }
28079     },
28080
28081     showMonthPicker : function(){
28082         this.createMonthPicker();
28083         var size = this.el.getSize();
28084         this.monthPicker.setSize(size);
28085         this.monthPicker.child('table').setSize(size);
28086
28087         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28088         this.updateMPMonth(this.mpSelMonth);
28089         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28090         this.updateMPYear(this.mpSelYear);
28091
28092         this.monthPicker.slideIn('t', {duration:.2});
28093     },
28094
28095     updateMPYear : function(y){
28096         this.mpyear = y;
28097         var ys = this.mpYears.elements;
28098         for(var i = 1; i <= 10; i++){
28099             var td = ys[i-1], y2;
28100             if((i%2) == 0){
28101                 y2 = y + Math.round(i * .5);
28102                 td.firstChild.innerHTML = y2;
28103                 td.xyear = y2;
28104             }else{
28105                 y2 = y - (5-Math.round(i * .5));
28106                 td.firstChild.innerHTML = y2;
28107                 td.xyear = y2;
28108             }
28109             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28110         }
28111     },
28112
28113     updateMPMonth : function(sm){
28114         this.mpMonths.each(function(m, a, i){
28115             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28116         });
28117     },
28118
28119     selectMPMonth: function(m){
28120         
28121     },
28122
28123     onMonthClick : function(e, t){
28124         e.stopEvent();
28125         var el = new Roo.Element(t), pn;
28126         if(el.is('button.x-date-mp-cancel')){
28127             this.hideMonthPicker();
28128         }
28129         else if(el.is('button.x-date-mp-ok')){
28130             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28131             this.hideMonthPicker();
28132         }
28133         else if(pn = el.up('td.x-date-mp-month', 2)){
28134             this.mpMonths.removeClass('x-date-mp-sel');
28135             pn.addClass('x-date-mp-sel');
28136             this.mpSelMonth = pn.dom.xmonth;
28137         }
28138         else if(pn = el.up('td.x-date-mp-year', 2)){
28139             this.mpYears.removeClass('x-date-mp-sel');
28140             pn.addClass('x-date-mp-sel');
28141             this.mpSelYear = pn.dom.xyear;
28142         }
28143         else if(el.is('a.x-date-mp-prev')){
28144             this.updateMPYear(this.mpyear-10);
28145         }
28146         else if(el.is('a.x-date-mp-next')){
28147             this.updateMPYear(this.mpyear+10);
28148         }
28149     },
28150
28151     onMonthDblClick : function(e, t){
28152         e.stopEvent();
28153         var el = new Roo.Element(t), pn;
28154         if(pn = el.up('td.x-date-mp-month', 2)){
28155             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28156             this.hideMonthPicker();
28157         }
28158         else if(pn = el.up('td.x-date-mp-year', 2)){
28159             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28160             this.hideMonthPicker();
28161         }
28162     },
28163
28164     hideMonthPicker : function(disableAnim){
28165         if(this.monthPicker){
28166             if(disableAnim === true){
28167                 this.monthPicker.hide();
28168             }else{
28169                 this.monthPicker.slideOut('t', {duration:.2});
28170             }
28171         }
28172     },
28173
28174     // private
28175     showPrevMonth : function(e){
28176         this.update(this.activeDate.add("mo", -1));
28177     },
28178
28179     // private
28180     showNextMonth : function(e){
28181         this.update(this.activeDate.add("mo", 1));
28182     },
28183
28184     // private
28185     showPrevYear : function(){
28186         this.update(this.activeDate.add("y", -1));
28187     },
28188
28189     // private
28190     showNextYear : function(){
28191         this.update(this.activeDate.add("y", 1));
28192     },
28193
28194     // private
28195     handleMouseWheel : function(e){
28196         var delta = e.getWheelDelta();
28197         if(delta > 0){
28198             this.showPrevMonth();
28199             e.stopEvent();
28200         } else if(delta < 0){
28201             this.showNextMonth();
28202             e.stopEvent();
28203         }
28204     },
28205
28206     // private
28207     handleDateClick : function(e, t){
28208         e.stopEvent();
28209         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28210             this.setValue(new Date(t.dateValue));
28211             this.fireEvent("select", this, this.value);
28212         }
28213     },
28214
28215     // private
28216     selectToday : function(){
28217         this.setValue(new Date().clearTime());
28218         this.fireEvent("select", this, this.value);
28219     },
28220
28221     // private
28222     update : function(date)
28223     {
28224         var vd = this.activeDate;
28225         this.activeDate = date;
28226         if(vd && this.el){
28227             var t = date.getTime();
28228             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28229                 this.cells.removeClass("x-date-selected");
28230                 this.cells.each(function(c){
28231                    if(c.dom.firstChild.dateValue == t){
28232                        c.addClass("x-date-selected");
28233                        setTimeout(function(){
28234                             try{c.dom.firstChild.focus();}catch(e){}
28235                        }, 50);
28236                        return false;
28237                    }
28238                 });
28239                 return;
28240             }
28241         }
28242         
28243         var days = date.getDaysInMonth();
28244         var firstOfMonth = date.getFirstDateOfMonth();
28245         var startingPos = firstOfMonth.getDay()-this.startDay;
28246
28247         if(startingPos <= this.startDay){
28248             startingPos += 7;
28249         }
28250
28251         var pm = date.add("mo", -1);
28252         var prevStart = pm.getDaysInMonth()-startingPos;
28253
28254         var cells = this.cells.elements;
28255         var textEls = this.textNodes;
28256         days += startingPos;
28257
28258         // convert everything to numbers so it's fast
28259         var day = 86400000;
28260         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28261         var today = new Date().clearTime().getTime();
28262         var sel = date.clearTime().getTime();
28263         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28264         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28265         var ddMatch = this.disabledDatesRE;
28266         var ddText = this.disabledDatesText;
28267         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28268         var ddaysText = this.disabledDaysText;
28269         var format = this.format;
28270
28271         var setCellClass = function(cal, cell){
28272             cell.title = "";
28273             var t = d.getTime();
28274             cell.firstChild.dateValue = t;
28275             if(t == today){
28276                 cell.className += " x-date-today";
28277                 cell.title = cal.todayText;
28278             }
28279             if(t == sel){
28280                 cell.className += " x-date-selected";
28281                 setTimeout(function(){
28282                     try{cell.firstChild.focus();}catch(e){}
28283                 }, 50);
28284             }
28285             // disabling
28286             if(t < min) {
28287                 cell.className = " x-date-disabled";
28288                 cell.title = cal.minText;
28289                 return;
28290             }
28291             if(t > max) {
28292                 cell.className = " x-date-disabled";
28293                 cell.title = cal.maxText;
28294                 return;
28295             }
28296             if(ddays){
28297                 if(ddays.indexOf(d.getDay()) != -1){
28298                     cell.title = ddaysText;
28299                     cell.className = " x-date-disabled";
28300                 }
28301             }
28302             if(ddMatch && format){
28303                 var fvalue = d.dateFormat(format);
28304                 if(ddMatch.test(fvalue)){
28305                     cell.title = ddText.replace("%0", fvalue);
28306                     cell.className = " x-date-disabled";
28307                 }
28308             }
28309         };
28310
28311         var i = 0;
28312         for(; i < startingPos; i++) {
28313             textEls[i].innerHTML = (++prevStart);
28314             d.setDate(d.getDate()+1);
28315             cells[i].className = "x-date-prevday";
28316             setCellClass(this, cells[i]);
28317         }
28318         for(; i < days; i++){
28319             intDay = i - startingPos + 1;
28320             textEls[i].innerHTML = (intDay);
28321             d.setDate(d.getDate()+1);
28322             cells[i].className = "x-date-active";
28323             setCellClass(this, cells[i]);
28324         }
28325         var extraDays = 0;
28326         for(; i < 42; i++) {
28327              textEls[i].innerHTML = (++extraDays);
28328              d.setDate(d.getDate()+1);
28329              cells[i].className = "x-date-nextday";
28330              setCellClass(this, cells[i]);
28331         }
28332
28333         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28334         this.fireEvent('monthchange', this, date);
28335         
28336         if(!this.internalRender){
28337             var main = this.el.dom.firstChild;
28338             var w = main.offsetWidth;
28339             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28340             Roo.fly(main).setWidth(w);
28341             this.internalRender = true;
28342             // opera does not respect the auto grow header center column
28343             // then, after it gets a width opera refuses to recalculate
28344             // without a second pass
28345             if(Roo.isOpera && !this.secondPass){
28346                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28347                 this.secondPass = true;
28348                 this.update.defer(10, this, [date]);
28349             }
28350         }
28351         
28352         
28353     }
28354 });        /*
28355  * Based on:
28356  * Ext JS Library 1.1.1
28357  * Copyright(c) 2006-2007, Ext JS, LLC.
28358  *
28359  * Originally Released Under LGPL - original licence link has changed is not relivant.
28360  *
28361  * Fork - LGPL
28362  * <script type="text/javascript">
28363  */
28364 /**
28365  * @class Roo.TabPanel
28366  * @extends Roo.util.Observable
28367  * A lightweight tab container.
28368  * <br><br>
28369  * Usage:
28370  * <pre><code>
28371 // basic tabs 1, built from existing content
28372 var tabs = new Roo.TabPanel("tabs1");
28373 tabs.addTab("script", "View Script");
28374 tabs.addTab("markup", "View Markup");
28375 tabs.activate("script");
28376
28377 // more advanced tabs, built from javascript
28378 var jtabs = new Roo.TabPanel("jtabs");
28379 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28380
28381 // set up the UpdateManager
28382 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28383 var updater = tab2.getUpdateManager();
28384 updater.setDefaultUrl("ajax1.htm");
28385 tab2.on('activate', updater.refresh, updater, true);
28386
28387 // Use setUrl for Ajax loading
28388 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28389 tab3.setUrl("ajax2.htm", null, true);
28390
28391 // Disabled tab
28392 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28393 tab4.disable();
28394
28395 jtabs.activate("jtabs-1");
28396  * </code></pre>
28397  * @constructor
28398  * Create a new TabPanel.
28399  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28400  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28401  */
28402 Roo.TabPanel = function(container, config){
28403     /**
28404     * The container element for this TabPanel.
28405     * @type Roo.Element
28406     */
28407     this.el = Roo.get(container, true);
28408     if(config){
28409         if(typeof config == "boolean"){
28410             this.tabPosition = config ? "bottom" : "top";
28411         }else{
28412             Roo.apply(this, config);
28413         }
28414     }
28415     if(this.tabPosition == "bottom"){
28416         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28417         this.el.addClass("x-tabs-bottom");
28418     }
28419     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28420     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28421     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28422     if(Roo.isIE){
28423         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28424     }
28425     if(this.tabPosition != "bottom"){
28426         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28427          * @type Roo.Element
28428          */
28429         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28430         this.el.addClass("x-tabs-top");
28431     }
28432     this.items = [];
28433
28434     this.bodyEl.setStyle("position", "relative");
28435
28436     this.active = null;
28437     this.activateDelegate = this.activate.createDelegate(this);
28438
28439     this.addEvents({
28440         /**
28441          * @event tabchange
28442          * Fires when the active tab changes
28443          * @param {Roo.TabPanel} this
28444          * @param {Roo.TabPanelItem} activePanel The new active tab
28445          */
28446         "tabchange": true,
28447         /**
28448          * @event beforetabchange
28449          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28450          * @param {Roo.TabPanel} this
28451          * @param {Object} e Set cancel to true on this object to cancel the tab change
28452          * @param {Roo.TabPanelItem} tab The tab being changed to
28453          */
28454         "beforetabchange" : true
28455     });
28456
28457     Roo.EventManager.onWindowResize(this.onResize, this);
28458     this.cpad = this.el.getPadding("lr");
28459     this.hiddenCount = 0;
28460
28461
28462     // toolbar on the tabbar support...
28463     if (this.toolbar) {
28464         var tcfg = this.toolbar;
28465         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28466         this.toolbar = new Roo.Toolbar(tcfg);
28467         if (Roo.isSafari) {
28468             var tbl = tcfg.container.child('table', true);
28469             tbl.setAttribute('width', '100%');
28470         }
28471         
28472     }
28473    
28474
28475
28476     Roo.TabPanel.superclass.constructor.call(this);
28477 };
28478
28479 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28480     /*
28481      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28482      */
28483     tabPosition : "top",
28484     /*
28485      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28486      */
28487     currentTabWidth : 0,
28488     /*
28489      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28490      */
28491     minTabWidth : 40,
28492     /*
28493      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28494      */
28495     maxTabWidth : 250,
28496     /*
28497      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28498      */
28499     preferredTabWidth : 175,
28500     /*
28501      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28502      */
28503     resizeTabs : false,
28504     /*
28505      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28506      */
28507     monitorResize : true,
28508     /*
28509      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28510      */
28511     toolbar : false,
28512
28513     /**
28514      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28515      * @param {String} id The id of the div to use <b>or create</b>
28516      * @param {String} text The text for the tab
28517      * @param {String} content (optional) Content to put in the TabPanelItem body
28518      * @param {Boolean} closable (optional) True to create a close icon on the tab
28519      * @return {Roo.TabPanelItem} The created TabPanelItem
28520      */
28521     addTab : function(id, text, content, closable){
28522         var item = new Roo.TabPanelItem(this, id, text, closable);
28523         this.addTabItem(item);
28524         if(content){
28525             item.setContent(content);
28526         }
28527         return item;
28528     },
28529
28530     /**
28531      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28532      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28533      * @return {Roo.TabPanelItem}
28534      */
28535     getTab : function(id){
28536         return this.items[id];
28537     },
28538
28539     /**
28540      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28541      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28542      */
28543     hideTab : function(id){
28544         var t = this.items[id];
28545         if(!t.isHidden()){
28546            t.setHidden(true);
28547            this.hiddenCount++;
28548            this.autoSizeTabs();
28549         }
28550     },
28551
28552     /**
28553      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28554      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28555      */
28556     unhideTab : function(id){
28557         var t = this.items[id];
28558         if(t.isHidden()){
28559            t.setHidden(false);
28560            this.hiddenCount--;
28561            this.autoSizeTabs();
28562         }
28563     },
28564
28565     /**
28566      * Adds an existing {@link Roo.TabPanelItem}.
28567      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28568      */
28569     addTabItem : function(item){
28570         this.items[item.id] = item;
28571         this.items.push(item);
28572         if(this.resizeTabs){
28573            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28574            this.autoSizeTabs();
28575         }else{
28576             item.autoSize();
28577         }
28578     },
28579
28580     /**
28581      * Removes a {@link Roo.TabPanelItem}.
28582      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28583      */
28584     removeTab : function(id){
28585         var items = this.items;
28586         var tab = items[id];
28587         if(!tab) { return; }
28588         var index = items.indexOf(tab);
28589         if(this.active == tab && items.length > 1){
28590             var newTab = this.getNextAvailable(index);
28591             if(newTab) {
28592                 newTab.activate();
28593             }
28594         }
28595         this.stripEl.dom.removeChild(tab.pnode.dom);
28596         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28597             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28598         }
28599         items.splice(index, 1);
28600         delete this.items[tab.id];
28601         tab.fireEvent("close", tab);
28602         tab.purgeListeners();
28603         this.autoSizeTabs();
28604     },
28605
28606     getNextAvailable : function(start){
28607         var items = this.items;
28608         var index = start;
28609         // look for a next tab that will slide over to
28610         // replace the one being removed
28611         while(index < items.length){
28612             var item = items[++index];
28613             if(item && !item.isHidden()){
28614                 return item;
28615             }
28616         }
28617         // if one isn't found select the previous tab (on the left)
28618         index = start;
28619         while(index >= 0){
28620             var item = items[--index];
28621             if(item && !item.isHidden()){
28622                 return item;
28623             }
28624         }
28625         return null;
28626     },
28627
28628     /**
28629      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28630      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28631      */
28632     disableTab : function(id){
28633         var tab = this.items[id];
28634         if(tab && this.active != tab){
28635             tab.disable();
28636         }
28637     },
28638
28639     /**
28640      * Enables a {@link Roo.TabPanelItem} that is disabled.
28641      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28642      */
28643     enableTab : function(id){
28644         var tab = this.items[id];
28645         tab.enable();
28646     },
28647
28648     /**
28649      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28650      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28651      * @return {Roo.TabPanelItem} The TabPanelItem.
28652      */
28653     activate : function(id){
28654         var tab = this.items[id];
28655         if(!tab){
28656             return null;
28657         }
28658         if(tab == this.active || tab.disabled){
28659             return tab;
28660         }
28661         var e = {};
28662         this.fireEvent("beforetabchange", this, e, tab);
28663         if(e.cancel !== true && !tab.disabled){
28664             if(this.active){
28665                 this.active.hide();
28666             }
28667             this.active = this.items[id];
28668             this.active.show();
28669             this.fireEvent("tabchange", this, this.active);
28670         }
28671         return tab;
28672     },
28673
28674     /**
28675      * Gets the active {@link Roo.TabPanelItem}.
28676      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28677      */
28678     getActiveTab : function(){
28679         return this.active;
28680     },
28681
28682     /**
28683      * Updates the tab body element to fit the height of the container element
28684      * for overflow scrolling
28685      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28686      */
28687     syncHeight : function(targetHeight){
28688         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28689         var bm = this.bodyEl.getMargins();
28690         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28691         this.bodyEl.setHeight(newHeight);
28692         return newHeight;
28693     },
28694
28695     onResize : function(){
28696         if(this.monitorResize){
28697             this.autoSizeTabs();
28698         }
28699     },
28700
28701     /**
28702      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28703      */
28704     beginUpdate : function(){
28705         this.updating = true;
28706     },
28707
28708     /**
28709      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28710      */
28711     endUpdate : function(){
28712         this.updating = false;
28713         this.autoSizeTabs();
28714     },
28715
28716     /**
28717      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28718      */
28719     autoSizeTabs : function(){
28720         var count = this.items.length;
28721         var vcount = count - this.hiddenCount;
28722         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28723             return;
28724         }
28725         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28726         var availWidth = Math.floor(w / vcount);
28727         var b = this.stripBody;
28728         if(b.getWidth() > w){
28729             var tabs = this.items;
28730             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28731             if(availWidth < this.minTabWidth){
28732                 /*if(!this.sleft){    // incomplete scrolling code
28733                     this.createScrollButtons();
28734                 }
28735                 this.showScroll();
28736                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28737             }
28738         }else{
28739             if(this.currentTabWidth < this.preferredTabWidth){
28740                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28741             }
28742         }
28743     },
28744
28745     /**
28746      * Returns the number of tabs in this TabPanel.
28747      * @return {Number}
28748      */
28749      getCount : function(){
28750          return this.items.length;
28751      },
28752
28753     /**
28754      * Resizes all the tabs to the passed width
28755      * @param {Number} The new width
28756      */
28757     setTabWidth : function(width){
28758         this.currentTabWidth = width;
28759         for(var i = 0, len = this.items.length; i < len; i++) {
28760                 if(!this.items[i].isHidden()) {
28761                 this.items[i].setWidth(width);
28762             }
28763         }
28764     },
28765
28766     /**
28767      * Destroys this TabPanel
28768      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28769      */
28770     destroy : function(removeEl){
28771         Roo.EventManager.removeResizeListener(this.onResize, this);
28772         for(var i = 0, len = this.items.length; i < len; i++){
28773             this.items[i].purgeListeners();
28774         }
28775         if(removeEl === true){
28776             this.el.update("");
28777             this.el.remove();
28778         }
28779     }
28780 });
28781
28782 /**
28783  * @class Roo.TabPanelItem
28784  * @extends Roo.util.Observable
28785  * Represents an individual item (tab plus body) in a TabPanel.
28786  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28787  * @param {String} id The id of this TabPanelItem
28788  * @param {String} text The text for the tab of this TabPanelItem
28789  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28790  */
28791 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28792     /**
28793      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28794      * @type Roo.TabPanel
28795      */
28796     this.tabPanel = tabPanel;
28797     /**
28798      * The id for this TabPanelItem
28799      * @type String
28800      */
28801     this.id = id;
28802     /** @private */
28803     this.disabled = false;
28804     /** @private */
28805     this.text = text;
28806     /** @private */
28807     this.loaded = false;
28808     this.closable = closable;
28809
28810     /**
28811      * The body element for this TabPanelItem.
28812      * @type Roo.Element
28813      */
28814     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28815     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28816     this.bodyEl.setStyle("display", "block");
28817     this.bodyEl.setStyle("zoom", "1");
28818     this.hideAction();
28819
28820     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28821     /** @private */
28822     this.el = Roo.get(els.el, true);
28823     this.inner = Roo.get(els.inner, true);
28824     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28825     this.pnode = Roo.get(els.el.parentNode, true);
28826     this.el.on("mousedown", this.onTabMouseDown, this);
28827     this.el.on("click", this.onTabClick, this);
28828     /** @private */
28829     if(closable){
28830         var c = Roo.get(els.close, true);
28831         c.dom.title = this.closeText;
28832         c.addClassOnOver("close-over");
28833         c.on("click", this.closeClick, this);
28834      }
28835
28836     this.addEvents({
28837          /**
28838          * @event activate
28839          * Fires when this tab becomes the active tab.
28840          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28841          * @param {Roo.TabPanelItem} this
28842          */
28843         "activate": true,
28844         /**
28845          * @event beforeclose
28846          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28847          * @param {Roo.TabPanelItem} this
28848          * @param {Object} e Set cancel to true on this object to cancel the close.
28849          */
28850         "beforeclose": true,
28851         /**
28852          * @event close
28853          * Fires when this tab is closed.
28854          * @param {Roo.TabPanelItem} this
28855          */
28856          "close": true,
28857         /**
28858          * @event deactivate
28859          * Fires when this tab is no longer the active tab.
28860          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28861          * @param {Roo.TabPanelItem} this
28862          */
28863          "deactivate" : true
28864     });
28865     this.hidden = false;
28866
28867     Roo.TabPanelItem.superclass.constructor.call(this);
28868 };
28869
28870 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28871     purgeListeners : function(){
28872        Roo.util.Observable.prototype.purgeListeners.call(this);
28873        this.el.removeAllListeners();
28874     },
28875     /**
28876      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28877      */
28878     show : function(){
28879         this.pnode.addClass("on");
28880         this.showAction();
28881         if(Roo.isOpera){
28882             this.tabPanel.stripWrap.repaint();
28883         }
28884         this.fireEvent("activate", this.tabPanel, this);
28885     },
28886
28887     /**
28888      * Returns true if this tab is the active tab.
28889      * @return {Boolean}
28890      */
28891     isActive : function(){
28892         return this.tabPanel.getActiveTab() == this;
28893     },
28894
28895     /**
28896      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28897      */
28898     hide : function(){
28899         this.pnode.removeClass("on");
28900         this.hideAction();
28901         this.fireEvent("deactivate", this.tabPanel, this);
28902     },
28903
28904     hideAction : function(){
28905         this.bodyEl.hide();
28906         this.bodyEl.setStyle("position", "absolute");
28907         this.bodyEl.setLeft("-20000px");
28908         this.bodyEl.setTop("-20000px");
28909     },
28910
28911     showAction : function(){
28912         this.bodyEl.setStyle("position", "relative");
28913         this.bodyEl.setTop("");
28914         this.bodyEl.setLeft("");
28915         this.bodyEl.show();
28916     },
28917
28918     /**
28919      * Set the tooltip for the tab.
28920      * @param {String} tooltip The tab's tooltip
28921      */
28922     setTooltip : function(text){
28923         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28924             this.textEl.dom.qtip = text;
28925             this.textEl.dom.removeAttribute('title');
28926         }else{
28927             this.textEl.dom.title = text;
28928         }
28929     },
28930
28931     onTabClick : function(e){
28932         e.preventDefault();
28933         this.tabPanel.activate(this.id);
28934     },
28935
28936     onTabMouseDown : function(e){
28937         e.preventDefault();
28938         this.tabPanel.activate(this.id);
28939     },
28940
28941     getWidth : function(){
28942         return this.inner.getWidth();
28943     },
28944
28945     setWidth : function(width){
28946         var iwidth = width - this.pnode.getPadding("lr");
28947         this.inner.setWidth(iwidth);
28948         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28949         this.pnode.setWidth(width);
28950     },
28951
28952     /**
28953      * Show or hide the tab
28954      * @param {Boolean} hidden True to hide or false to show.
28955      */
28956     setHidden : function(hidden){
28957         this.hidden = hidden;
28958         this.pnode.setStyle("display", hidden ? "none" : "");
28959     },
28960
28961     /**
28962      * Returns true if this tab is "hidden"
28963      * @return {Boolean}
28964      */
28965     isHidden : function(){
28966         return this.hidden;
28967     },
28968
28969     /**
28970      * Returns the text for this tab
28971      * @return {String}
28972      */
28973     getText : function(){
28974         return this.text;
28975     },
28976
28977     autoSize : function(){
28978         //this.el.beginMeasure();
28979         this.textEl.setWidth(1);
28980         /*
28981          *  #2804 [new] Tabs in Roojs
28982          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28983          */
28984         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28985         //this.el.endMeasure();
28986     },
28987
28988     /**
28989      * Sets the text for the tab (Note: this also sets the tooltip text)
28990      * @param {String} text The tab's text and tooltip
28991      */
28992     setText : function(text){
28993         this.text = text;
28994         this.textEl.update(text);
28995         this.setTooltip(text);
28996         if(!this.tabPanel.resizeTabs){
28997             this.autoSize();
28998         }
28999     },
29000     /**
29001      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
29002      */
29003     activate : function(){
29004         this.tabPanel.activate(this.id);
29005     },
29006
29007     /**
29008      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
29009      */
29010     disable : function(){
29011         if(this.tabPanel.active != this){
29012             this.disabled = true;
29013             this.pnode.addClass("disabled");
29014         }
29015     },
29016
29017     /**
29018      * Enables this TabPanelItem if it was previously disabled.
29019      */
29020     enable : function(){
29021         this.disabled = false;
29022         this.pnode.removeClass("disabled");
29023     },
29024
29025     /**
29026      * Sets the content for this TabPanelItem.
29027      * @param {String} content The content
29028      * @param {Boolean} loadScripts true to look for and load scripts
29029      */
29030     setContent : function(content, loadScripts){
29031         this.bodyEl.update(content, loadScripts);
29032     },
29033
29034     /**
29035      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29036      * @return {Roo.UpdateManager} The UpdateManager
29037      */
29038     getUpdateManager : function(){
29039         return this.bodyEl.getUpdateManager();
29040     },
29041
29042     /**
29043      * Set a URL to be used to load the content for this TabPanelItem.
29044      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29045      * @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)
29046      * @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)
29047      * @return {Roo.UpdateManager} The UpdateManager
29048      */
29049     setUrl : function(url, params, loadOnce){
29050         if(this.refreshDelegate){
29051             this.un('activate', this.refreshDelegate);
29052         }
29053         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29054         this.on("activate", this.refreshDelegate);
29055         return this.bodyEl.getUpdateManager();
29056     },
29057
29058     /** @private */
29059     _handleRefresh : function(url, params, loadOnce){
29060         if(!loadOnce || !this.loaded){
29061             var updater = this.bodyEl.getUpdateManager();
29062             updater.update(url, params, this._setLoaded.createDelegate(this));
29063         }
29064     },
29065
29066     /**
29067      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29068      *   Will fail silently if the setUrl method has not been called.
29069      *   This does not activate the panel, just updates its content.
29070      */
29071     refresh : function(){
29072         if(this.refreshDelegate){
29073            this.loaded = false;
29074            this.refreshDelegate();
29075         }
29076     },
29077
29078     /** @private */
29079     _setLoaded : function(){
29080         this.loaded = true;
29081     },
29082
29083     /** @private */
29084     closeClick : function(e){
29085         var o = {};
29086         e.stopEvent();
29087         this.fireEvent("beforeclose", this, o);
29088         if(o.cancel !== true){
29089             this.tabPanel.removeTab(this.id);
29090         }
29091     },
29092     /**
29093      * The text displayed in the tooltip for the close icon.
29094      * @type String
29095      */
29096     closeText : "Close this tab"
29097 });
29098
29099 /** @private */
29100 Roo.TabPanel.prototype.createStrip = function(container){
29101     var strip = document.createElement("div");
29102     strip.className = "x-tabs-wrap";
29103     container.appendChild(strip);
29104     return strip;
29105 };
29106 /** @private */
29107 Roo.TabPanel.prototype.createStripList = function(strip){
29108     // div wrapper for retard IE
29109     // returns the "tr" element.
29110     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29111         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29112         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29113     return strip.firstChild.firstChild.firstChild.firstChild;
29114 };
29115 /** @private */
29116 Roo.TabPanel.prototype.createBody = function(container){
29117     var body = document.createElement("div");
29118     Roo.id(body, "tab-body");
29119     Roo.fly(body).addClass("x-tabs-body");
29120     container.appendChild(body);
29121     return body;
29122 };
29123 /** @private */
29124 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29125     var body = Roo.getDom(id);
29126     if(!body){
29127         body = document.createElement("div");
29128         body.id = id;
29129     }
29130     Roo.fly(body).addClass("x-tabs-item-body");
29131     bodyEl.insertBefore(body, bodyEl.firstChild);
29132     return body;
29133 };
29134 /** @private */
29135 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29136     var td = document.createElement("td");
29137     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29138     //stripEl.appendChild(td);
29139     if(closable){
29140         td.className = "x-tabs-closable";
29141         if(!this.closeTpl){
29142             this.closeTpl = new Roo.Template(
29143                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29144                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29145                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29146             );
29147         }
29148         var el = this.closeTpl.overwrite(td, {"text": text});
29149         var close = el.getElementsByTagName("div")[0];
29150         var inner = el.getElementsByTagName("em")[0];
29151         return {"el": el, "close": close, "inner": inner};
29152     } else {
29153         if(!this.tabTpl){
29154             this.tabTpl = new Roo.Template(
29155                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29156                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29157             );
29158         }
29159         var el = this.tabTpl.overwrite(td, {"text": text});
29160         var inner = el.getElementsByTagName("em")[0];
29161         return {"el": el, "inner": inner};
29162     }
29163 };/*
29164  * Based on:
29165  * Ext JS Library 1.1.1
29166  * Copyright(c) 2006-2007, Ext JS, LLC.
29167  *
29168  * Originally Released Under LGPL - original licence link has changed is not relivant.
29169  *
29170  * Fork - LGPL
29171  * <script type="text/javascript">
29172  */
29173
29174 /**
29175  * @class Roo.Button
29176  * @extends Roo.util.Observable
29177  * Simple Button class
29178  * @cfg {String} text The button text
29179  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29180  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29181  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29182  * @cfg {Object} scope The scope of the handler
29183  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29184  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29185  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29186  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29187  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29188  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29189    applies if enableToggle = true)
29190  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29191  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29192   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29193  * @constructor
29194  * Create a new button
29195  * @param {Object} config The config object
29196  */
29197 Roo.Button = function(renderTo, config)
29198 {
29199     if (!config) {
29200         config = renderTo;
29201         renderTo = config.renderTo || false;
29202     }
29203     
29204     Roo.apply(this, config);
29205     this.addEvents({
29206         /**
29207              * @event click
29208              * Fires when this button is clicked
29209              * @param {Button} this
29210              * @param {EventObject} e The click event
29211              */
29212             "click" : true,
29213         /**
29214              * @event toggle
29215              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29216              * @param {Button} this
29217              * @param {Boolean} pressed
29218              */
29219             "toggle" : true,
29220         /**
29221              * @event mouseover
29222              * Fires when the mouse hovers over the button
29223              * @param {Button} this
29224              * @param {Event} e The event object
29225              */
29226         'mouseover' : true,
29227         /**
29228              * @event mouseout
29229              * Fires when the mouse exits the button
29230              * @param {Button} this
29231              * @param {Event} e The event object
29232              */
29233         'mouseout': true,
29234          /**
29235              * @event render
29236              * Fires when the button is rendered
29237              * @param {Button} this
29238              */
29239         'render': true
29240     });
29241     if(this.menu){
29242         this.menu = Roo.menu.MenuMgr.get(this.menu);
29243     }
29244     // register listeners first!!  - so render can be captured..
29245     Roo.util.Observable.call(this);
29246     if(renderTo){
29247         this.render(renderTo);
29248     }
29249     
29250   
29251 };
29252
29253 Roo.extend(Roo.Button, Roo.util.Observable, {
29254     /**
29255      * 
29256      */
29257     
29258     /**
29259      * Read-only. True if this button is hidden
29260      * @type Boolean
29261      */
29262     hidden : false,
29263     /**
29264      * Read-only. True if this button is disabled
29265      * @type Boolean
29266      */
29267     disabled : false,
29268     /**
29269      * Read-only. True if this button is pressed (only if enableToggle = true)
29270      * @type Boolean
29271      */
29272     pressed : false,
29273
29274     /**
29275      * @cfg {Number} tabIndex 
29276      * The DOM tabIndex for this button (defaults to undefined)
29277      */
29278     tabIndex : undefined,
29279
29280     /**
29281      * @cfg {Boolean} enableToggle
29282      * True to enable pressed/not pressed toggling (defaults to false)
29283      */
29284     enableToggle: false,
29285     /**
29286      * @cfg {Mixed} menu
29287      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29288      */
29289     menu : undefined,
29290     /**
29291      * @cfg {String} menuAlign
29292      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29293      */
29294     menuAlign : "tl-bl?",
29295
29296     /**
29297      * @cfg {String} iconCls
29298      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29299      */
29300     iconCls : undefined,
29301     /**
29302      * @cfg {String} type
29303      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29304      */
29305     type : 'button',
29306
29307     // private
29308     menuClassTarget: 'tr',
29309
29310     /**
29311      * @cfg {String} clickEvent
29312      * The type of event to map to the button's event handler (defaults to 'click')
29313      */
29314     clickEvent : 'click',
29315
29316     /**
29317      * @cfg {Boolean} handleMouseEvents
29318      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29319      */
29320     handleMouseEvents : true,
29321
29322     /**
29323      * @cfg {String} tooltipType
29324      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29325      */
29326     tooltipType : 'qtip',
29327
29328     /**
29329      * @cfg {String} cls
29330      * A CSS class to apply to the button's main element.
29331      */
29332     
29333     /**
29334      * @cfg {Roo.Template} template (Optional)
29335      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29336      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29337      * require code modifications if required elements (e.g. a button) aren't present.
29338      */
29339
29340     // private
29341     render : function(renderTo){
29342         var btn;
29343         if(this.hideParent){
29344             this.parentEl = Roo.get(renderTo);
29345         }
29346         if(!this.dhconfig){
29347             if(!this.template){
29348                 if(!Roo.Button.buttonTemplate){
29349                     // hideous table template
29350                     Roo.Button.buttonTemplate = new Roo.Template(
29351                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29352                         '<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>',
29353                         "</tr></tbody></table>");
29354                 }
29355                 this.template = Roo.Button.buttonTemplate;
29356             }
29357             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29358             var btnEl = btn.child("button:first");
29359             btnEl.on('focus', this.onFocus, this);
29360             btnEl.on('blur', this.onBlur, this);
29361             if(this.cls){
29362                 btn.addClass(this.cls);
29363             }
29364             if(this.icon){
29365                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29366             }
29367             if(this.iconCls){
29368                 btnEl.addClass(this.iconCls);
29369                 if(!this.cls){
29370                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29371                 }
29372             }
29373             if(this.tabIndex !== undefined){
29374                 btnEl.dom.tabIndex = this.tabIndex;
29375             }
29376             if(this.tooltip){
29377                 if(typeof this.tooltip == 'object'){
29378                     Roo.QuickTips.tips(Roo.apply({
29379                           target: btnEl.id
29380                     }, this.tooltip));
29381                 } else {
29382                     btnEl.dom[this.tooltipType] = this.tooltip;
29383                 }
29384             }
29385         }else{
29386             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29387         }
29388         this.el = btn;
29389         if(this.id){
29390             this.el.dom.id = this.el.id = this.id;
29391         }
29392         if(this.menu){
29393             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29394             this.menu.on("show", this.onMenuShow, this);
29395             this.menu.on("hide", this.onMenuHide, this);
29396         }
29397         btn.addClass("x-btn");
29398         if(Roo.isIE && !Roo.isIE7){
29399             this.autoWidth.defer(1, this);
29400         }else{
29401             this.autoWidth();
29402         }
29403         if(this.handleMouseEvents){
29404             btn.on("mouseover", this.onMouseOver, this);
29405             btn.on("mouseout", this.onMouseOut, this);
29406             btn.on("mousedown", this.onMouseDown, this);
29407         }
29408         btn.on(this.clickEvent, this.onClick, this);
29409         //btn.on("mouseup", this.onMouseUp, this);
29410         if(this.hidden){
29411             this.hide();
29412         }
29413         if(this.disabled){
29414             this.disable();
29415         }
29416         Roo.ButtonToggleMgr.register(this);
29417         if(this.pressed){
29418             this.el.addClass("x-btn-pressed");
29419         }
29420         if(this.repeat){
29421             var repeater = new Roo.util.ClickRepeater(btn,
29422                 typeof this.repeat == "object" ? this.repeat : {}
29423             );
29424             repeater.on("click", this.onClick,  this);
29425         }
29426         
29427         this.fireEvent('render', this);
29428         
29429     },
29430     /**
29431      * Returns the button's underlying element
29432      * @return {Roo.Element} The element
29433      */
29434     getEl : function(){
29435         return this.el;  
29436     },
29437     
29438     /**
29439      * Destroys this Button and removes any listeners.
29440      */
29441     destroy : function(){
29442         Roo.ButtonToggleMgr.unregister(this);
29443         this.el.removeAllListeners();
29444         this.purgeListeners();
29445         this.el.remove();
29446     },
29447
29448     // private
29449     autoWidth : function(){
29450         if(this.el){
29451             this.el.setWidth("auto");
29452             if(Roo.isIE7 && Roo.isStrict){
29453                 var ib = this.el.child('button');
29454                 if(ib && ib.getWidth() > 20){
29455                     ib.clip();
29456                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29457                 }
29458             }
29459             if(this.minWidth){
29460                 if(this.hidden){
29461                     this.el.beginMeasure();
29462                 }
29463                 if(this.el.getWidth() < this.minWidth){
29464                     this.el.setWidth(this.minWidth);
29465                 }
29466                 if(this.hidden){
29467                     this.el.endMeasure();
29468                 }
29469             }
29470         }
29471     },
29472
29473     /**
29474      * Assigns this button's click handler
29475      * @param {Function} handler The function to call when the button is clicked
29476      * @param {Object} scope (optional) Scope for the function passed in
29477      */
29478     setHandler : function(handler, scope){
29479         this.handler = handler;
29480         this.scope = scope;  
29481     },
29482     
29483     /**
29484      * Sets this button's text
29485      * @param {String} text The button text
29486      */
29487     setText : function(text){
29488         this.text = text;
29489         if(this.el){
29490             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29491         }
29492         this.autoWidth();
29493     },
29494     
29495     /**
29496      * Gets the text for this button
29497      * @return {String} The button text
29498      */
29499     getText : function(){
29500         return this.text;  
29501     },
29502     
29503     /**
29504      * Show this button
29505      */
29506     show: function(){
29507         this.hidden = false;
29508         if(this.el){
29509             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29510         }
29511     },
29512     
29513     /**
29514      * Hide this button
29515      */
29516     hide: function(){
29517         this.hidden = true;
29518         if(this.el){
29519             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29520         }
29521     },
29522     
29523     /**
29524      * Convenience function for boolean show/hide
29525      * @param {Boolean} visible True to show, false to hide
29526      */
29527     setVisible: function(visible){
29528         if(visible) {
29529             this.show();
29530         }else{
29531             this.hide();
29532         }
29533     },
29534     
29535     /**
29536      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29537      * @param {Boolean} state (optional) Force a particular state
29538      */
29539     toggle : function(state){
29540         state = state === undefined ? !this.pressed : state;
29541         if(state != this.pressed){
29542             if(state){
29543                 this.el.addClass("x-btn-pressed");
29544                 this.pressed = true;
29545                 this.fireEvent("toggle", this, true);
29546             }else{
29547                 this.el.removeClass("x-btn-pressed");
29548                 this.pressed = false;
29549                 this.fireEvent("toggle", this, false);
29550             }
29551             if(this.toggleHandler){
29552                 this.toggleHandler.call(this.scope || this, this, state);
29553             }
29554         }
29555     },
29556     
29557     /**
29558      * Focus the button
29559      */
29560     focus : function(){
29561         this.el.child('button:first').focus();
29562     },
29563     
29564     /**
29565      * Disable this button
29566      */
29567     disable : function(){
29568         if(this.el){
29569             this.el.addClass("x-btn-disabled");
29570         }
29571         this.disabled = true;
29572     },
29573     
29574     /**
29575      * Enable this button
29576      */
29577     enable : function(){
29578         if(this.el){
29579             this.el.removeClass("x-btn-disabled");
29580         }
29581         this.disabled = false;
29582     },
29583
29584     /**
29585      * Convenience function for boolean enable/disable
29586      * @param {Boolean} enabled True to enable, false to disable
29587      */
29588     setDisabled : function(v){
29589         this[v !== true ? "enable" : "disable"]();
29590     },
29591
29592     // private
29593     onClick : function(e)
29594     {
29595         if(e){
29596             e.preventDefault();
29597         }
29598         if(e.button != 0){
29599             return;
29600         }
29601         if(!this.disabled){
29602             if(this.enableToggle){
29603                 this.toggle();
29604             }
29605             if(this.menu && !this.menu.isVisible()){
29606                 this.menu.show(this.el, this.menuAlign);
29607             }
29608             this.fireEvent("click", this, e);
29609             if(this.handler){
29610                 this.el.removeClass("x-btn-over");
29611                 this.handler.call(this.scope || this, this, e);
29612             }
29613         }
29614     },
29615     // private
29616     onMouseOver : function(e){
29617         if(!this.disabled){
29618             this.el.addClass("x-btn-over");
29619             this.fireEvent('mouseover', this, e);
29620         }
29621     },
29622     // private
29623     onMouseOut : function(e){
29624         if(!e.within(this.el,  true)){
29625             this.el.removeClass("x-btn-over");
29626             this.fireEvent('mouseout', this, e);
29627         }
29628     },
29629     // private
29630     onFocus : function(e){
29631         if(!this.disabled){
29632             this.el.addClass("x-btn-focus");
29633         }
29634     },
29635     // private
29636     onBlur : function(e){
29637         this.el.removeClass("x-btn-focus");
29638     },
29639     // private
29640     onMouseDown : function(e){
29641         if(!this.disabled && e.button == 0){
29642             this.el.addClass("x-btn-click");
29643             Roo.get(document).on('mouseup', this.onMouseUp, this);
29644         }
29645     },
29646     // private
29647     onMouseUp : function(e){
29648         if(e.button == 0){
29649             this.el.removeClass("x-btn-click");
29650             Roo.get(document).un('mouseup', this.onMouseUp, this);
29651         }
29652     },
29653     // private
29654     onMenuShow : function(e){
29655         this.el.addClass("x-btn-menu-active");
29656     },
29657     // private
29658     onMenuHide : function(e){
29659         this.el.removeClass("x-btn-menu-active");
29660     }   
29661 });
29662
29663 // Private utility class used by Button
29664 Roo.ButtonToggleMgr = function(){
29665    var groups = {};
29666    
29667    function toggleGroup(btn, state){
29668        if(state){
29669            var g = groups[btn.toggleGroup];
29670            for(var i = 0, l = g.length; i < l; i++){
29671                if(g[i] != btn){
29672                    g[i].toggle(false);
29673                }
29674            }
29675        }
29676    }
29677    
29678    return {
29679        register : function(btn){
29680            if(!btn.toggleGroup){
29681                return;
29682            }
29683            var g = groups[btn.toggleGroup];
29684            if(!g){
29685                g = groups[btn.toggleGroup] = [];
29686            }
29687            g.push(btn);
29688            btn.on("toggle", toggleGroup);
29689        },
29690        
29691        unregister : function(btn){
29692            if(!btn.toggleGroup){
29693                return;
29694            }
29695            var g = groups[btn.toggleGroup];
29696            if(g){
29697                g.remove(btn);
29698                btn.un("toggle", toggleGroup);
29699            }
29700        }
29701    };
29702 }();/*
29703  * Based on:
29704  * Ext JS Library 1.1.1
29705  * Copyright(c) 2006-2007, Ext JS, LLC.
29706  *
29707  * Originally Released Under LGPL - original licence link has changed is not relivant.
29708  *
29709  * Fork - LGPL
29710  * <script type="text/javascript">
29711  */
29712  
29713 /**
29714  * @class Roo.SplitButton
29715  * @extends Roo.Button
29716  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29717  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29718  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29719  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29720  * @cfg {String} arrowTooltip The title attribute of the arrow
29721  * @constructor
29722  * Create a new menu button
29723  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29724  * @param {Object} config The config object
29725  */
29726 Roo.SplitButton = function(renderTo, config){
29727     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29728     /**
29729      * @event arrowclick
29730      * Fires when this button's arrow is clicked
29731      * @param {SplitButton} this
29732      * @param {EventObject} e The click event
29733      */
29734     this.addEvents({"arrowclick":true});
29735 };
29736
29737 Roo.extend(Roo.SplitButton, Roo.Button, {
29738     render : function(renderTo){
29739         // this is one sweet looking template!
29740         var tpl = new Roo.Template(
29741             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29742             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29743             '<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>',
29744             "</tbody></table></td><td>",
29745             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29746             '<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>',
29747             "</tbody></table></td></tr></table>"
29748         );
29749         var btn = tpl.append(renderTo, [this.text, this.type], true);
29750         var btnEl = btn.child("button");
29751         if(this.cls){
29752             btn.addClass(this.cls);
29753         }
29754         if(this.icon){
29755             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29756         }
29757         if(this.iconCls){
29758             btnEl.addClass(this.iconCls);
29759             if(!this.cls){
29760                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29761             }
29762         }
29763         this.el = btn;
29764         if(this.handleMouseEvents){
29765             btn.on("mouseover", this.onMouseOver, this);
29766             btn.on("mouseout", this.onMouseOut, this);
29767             btn.on("mousedown", this.onMouseDown, this);
29768             btn.on("mouseup", this.onMouseUp, this);
29769         }
29770         btn.on(this.clickEvent, this.onClick, this);
29771         if(this.tooltip){
29772             if(typeof this.tooltip == 'object'){
29773                 Roo.QuickTips.tips(Roo.apply({
29774                       target: btnEl.id
29775                 }, this.tooltip));
29776             } else {
29777                 btnEl.dom[this.tooltipType] = this.tooltip;
29778             }
29779         }
29780         if(this.arrowTooltip){
29781             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29782         }
29783         if(this.hidden){
29784             this.hide();
29785         }
29786         if(this.disabled){
29787             this.disable();
29788         }
29789         if(this.pressed){
29790             this.el.addClass("x-btn-pressed");
29791         }
29792         if(Roo.isIE && !Roo.isIE7){
29793             this.autoWidth.defer(1, this);
29794         }else{
29795             this.autoWidth();
29796         }
29797         if(this.menu){
29798             this.menu.on("show", this.onMenuShow, this);
29799             this.menu.on("hide", this.onMenuHide, this);
29800         }
29801         this.fireEvent('render', this);
29802     },
29803
29804     // private
29805     autoWidth : function(){
29806         if(this.el){
29807             var tbl = this.el.child("table:first");
29808             var tbl2 = this.el.child("table:last");
29809             this.el.setWidth("auto");
29810             tbl.setWidth("auto");
29811             if(Roo.isIE7 && Roo.isStrict){
29812                 var ib = this.el.child('button:first');
29813                 if(ib && ib.getWidth() > 20){
29814                     ib.clip();
29815                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29816                 }
29817             }
29818             if(this.minWidth){
29819                 if(this.hidden){
29820                     this.el.beginMeasure();
29821                 }
29822                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29823                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29824                 }
29825                 if(this.hidden){
29826                     this.el.endMeasure();
29827                 }
29828             }
29829             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29830         } 
29831     },
29832     /**
29833      * Sets this button's click handler
29834      * @param {Function} handler The function to call when the button is clicked
29835      * @param {Object} scope (optional) Scope for the function passed above
29836      */
29837     setHandler : function(handler, scope){
29838         this.handler = handler;
29839         this.scope = scope;  
29840     },
29841     
29842     /**
29843      * Sets this button's arrow click handler
29844      * @param {Function} handler The function to call when the arrow is clicked
29845      * @param {Object} scope (optional) Scope for the function passed above
29846      */
29847     setArrowHandler : function(handler, scope){
29848         this.arrowHandler = handler;
29849         this.scope = scope;  
29850     },
29851     
29852     /**
29853      * Focus the button
29854      */
29855     focus : function(){
29856         if(this.el){
29857             this.el.child("button:first").focus();
29858         }
29859     },
29860
29861     // private
29862     onClick : function(e){
29863         e.preventDefault();
29864         if(!this.disabled){
29865             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29866                 if(this.menu && !this.menu.isVisible()){
29867                     this.menu.show(this.el, this.menuAlign);
29868                 }
29869                 this.fireEvent("arrowclick", this, e);
29870                 if(this.arrowHandler){
29871                     this.arrowHandler.call(this.scope || this, this, e);
29872                 }
29873             }else{
29874                 this.fireEvent("click", this, e);
29875                 if(this.handler){
29876                     this.handler.call(this.scope || this, this, e);
29877                 }
29878             }
29879         }
29880     },
29881     // private
29882     onMouseDown : function(e){
29883         if(!this.disabled){
29884             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29885         }
29886     },
29887     // private
29888     onMouseUp : function(e){
29889         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29890     }   
29891 });
29892
29893
29894 // backwards compat
29895 Roo.MenuButton = Roo.SplitButton;/*
29896  * Based on:
29897  * Ext JS Library 1.1.1
29898  * Copyright(c) 2006-2007, Ext JS, LLC.
29899  *
29900  * Originally Released Under LGPL - original licence link has changed is not relivant.
29901  *
29902  * Fork - LGPL
29903  * <script type="text/javascript">
29904  */
29905
29906 /**
29907  * @class Roo.Toolbar
29908  * Basic Toolbar class.
29909  * @constructor
29910  * Creates a new Toolbar
29911  * @param {Object} container The config object
29912  */ 
29913 Roo.Toolbar = function(container, buttons, config)
29914 {
29915     /// old consturctor format still supported..
29916     if(container instanceof Array){ // omit the container for later rendering
29917         buttons = container;
29918         config = buttons;
29919         container = null;
29920     }
29921     if (typeof(container) == 'object' && container.xtype) {
29922         config = container;
29923         container = config.container;
29924         buttons = config.buttons || []; // not really - use items!!
29925     }
29926     var xitems = [];
29927     if (config && config.items) {
29928         xitems = config.items;
29929         delete config.items;
29930     }
29931     Roo.apply(this, config);
29932     this.buttons = buttons;
29933     
29934     if(container){
29935         this.render(container);
29936     }
29937     this.xitems = xitems;
29938     Roo.each(xitems, function(b) {
29939         this.add(b);
29940     }, this);
29941     
29942 };
29943
29944 Roo.Toolbar.prototype = {
29945     /**
29946      * @cfg {Array} items
29947      * array of button configs or elements to add (will be converted to a MixedCollection)
29948      */
29949     
29950     /**
29951      * @cfg {String/HTMLElement/Element} container
29952      * The id or element that will contain the toolbar
29953      */
29954     // private
29955     render : function(ct){
29956         this.el = Roo.get(ct);
29957         if(this.cls){
29958             this.el.addClass(this.cls);
29959         }
29960         // using a table allows for vertical alignment
29961         // 100% width is needed by Safari...
29962         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29963         this.tr = this.el.child("tr", true);
29964         var autoId = 0;
29965         this.items = new Roo.util.MixedCollection(false, function(o){
29966             return o.id || ("item" + (++autoId));
29967         });
29968         if(this.buttons){
29969             this.add.apply(this, this.buttons);
29970             delete this.buttons;
29971         }
29972     },
29973
29974     /**
29975      * Adds element(s) to the toolbar -- this function takes a variable number of 
29976      * arguments of mixed type and adds them to the toolbar.
29977      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29978      * <ul>
29979      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29980      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29981      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29982      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29983      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29984      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29985      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29986      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29987      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29988      * </ul>
29989      * @param {Mixed} arg2
29990      * @param {Mixed} etc.
29991      */
29992     add : function(){
29993         var a = arguments, l = a.length;
29994         for(var i = 0; i < l; i++){
29995             this._add(a[i]);
29996         }
29997     },
29998     // private..
29999     _add : function(el) {
30000         
30001         if (el.xtype) {
30002             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
30003         }
30004         
30005         if (el.applyTo){ // some kind of form field
30006             return this.addField(el);
30007         } 
30008         if (el.render){ // some kind of Toolbar.Item
30009             return this.addItem(el);
30010         }
30011         if (typeof el == "string"){ // string
30012             if(el == "separator" || el == "-"){
30013                 return this.addSeparator();
30014             }
30015             if (el == " "){
30016                 return this.addSpacer();
30017             }
30018             if(el == "->"){
30019                 return this.addFill();
30020             }
30021             return this.addText(el);
30022             
30023         }
30024         if(el.tagName){ // element
30025             return this.addElement(el);
30026         }
30027         if(typeof el == "object"){ // must be button config?
30028             return this.addButton(el);
30029         }
30030         // and now what?!?!
30031         return false;
30032         
30033     },
30034     
30035     /**
30036      * Add an Xtype element
30037      * @param {Object} xtype Xtype Object
30038      * @return {Object} created Object
30039      */
30040     addxtype : function(e){
30041         return this.add(e);  
30042     },
30043     
30044     /**
30045      * Returns the Element for this toolbar.
30046      * @return {Roo.Element}
30047      */
30048     getEl : function(){
30049         return this.el;  
30050     },
30051     
30052     /**
30053      * Adds a separator
30054      * @return {Roo.Toolbar.Item} The separator item
30055      */
30056     addSeparator : function(){
30057         return this.addItem(new Roo.Toolbar.Separator());
30058     },
30059
30060     /**
30061      * Adds a spacer element
30062      * @return {Roo.Toolbar.Spacer} The spacer item
30063      */
30064     addSpacer : function(){
30065         return this.addItem(new Roo.Toolbar.Spacer());
30066     },
30067
30068     /**
30069      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30070      * @return {Roo.Toolbar.Fill} The fill item
30071      */
30072     addFill : function(){
30073         return this.addItem(new Roo.Toolbar.Fill());
30074     },
30075
30076     /**
30077      * Adds any standard HTML element to the toolbar
30078      * @param {String/HTMLElement/Element} el The element or id of the element to add
30079      * @return {Roo.Toolbar.Item} The element's item
30080      */
30081     addElement : function(el){
30082         return this.addItem(new Roo.Toolbar.Item(el));
30083     },
30084     /**
30085      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30086      * @type Roo.util.MixedCollection  
30087      */
30088     items : false,
30089      
30090     /**
30091      * Adds any Toolbar.Item or subclass
30092      * @param {Roo.Toolbar.Item} item
30093      * @return {Roo.Toolbar.Item} The item
30094      */
30095     addItem : function(item){
30096         var td = this.nextBlock();
30097         item.render(td);
30098         this.items.add(item);
30099         return item;
30100     },
30101     
30102     /**
30103      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30104      * @param {Object/Array} config A button config or array of configs
30105      * @return {Roo.Toolbar.Button/Array}
30106      */
30107     addButton : function(config){
30108         if(config instanceof Array){
30109             var buttons = [];
30110             for(var i = 0, len = config.length; i < len; i++) {
30111                 buttons.push(this.addButton(config[i]));
30112             }
30113             return buttons;
30114         }
30115         var b = config;
30116         if(!(config instanceof Roo.Toolbar.Button)){
30117             b = config.split ?
30118                 new Roo.Toolbar.SplitButton(config) :
30119                 new Roo.Toolbar.Button(config);
30120         }
30121         var td = this.nextBlock();
30122         b.render(td);
30123         this.items.add(b);
30124         return b;
30125     },
30126     
30127     /**
30128      * Adds text to the toolbar
30129      * @param {String} text The text to add
30130      * @return {Roo.Toolbar.Item} The element's item
30131      */
30132     addText : function(text){
30133         return this.addItem(new Roo.Toolbar.TextItem(text));
30134     },
30135     
30136     /**
30137      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30138      * @param {Number} index The index where the item is to be inserted
30139      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30140      * @return {Roo.Toolbar.Button/Item}
30141      */
30142     insertButton : function(index, item){
30143         if(item instanceof Array){
30144             var buttons = [];
30145             for(var i = 0, len = item.length; i < len; i++) {
30146                buttons.push(this.insertButton(index + i, item[i]));
30147             }
30148             return buttons;
30149         }
30150         if (!(item instanceof Roo.Toolbar.Button)){
30151            item = new Roo.Toolbar.Button(item);
30152         }
30153         var td = document.createElement("td");
30154         this.tr.insertBefore(td, this.tr.childNodes[index]);
30155         item.render(td);
30156         this.items.insert(index, item);
30157         return item;
30158     },
30159     
30160     /**
30161      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30162      * @param {Object} config
30163      * @return {Roo.Toolbar.Item} The element's item
30164      */
30165     addDom : function(config, returnEl){
30166         var td = this.nextBlock();
30167         Roo.DomHelper.overwrite(td, config);
30168         var ti = new Roo.Toolbar.Item(td.firstChild);
30169         ti.render(td);
30170         this.items.add(ti);
30171         return ti;
30172     },
30173
30174     /**
30175      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30176      * @type Roo.util.MixedCollection  
30177      */
30178     fields : false,
30179     
30180     /**
30181      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30182      * Note: the field should not have been rendered yet. For a field that has already been
30183      * rendered, use {@link #addElement}.
30184      * @param {Roo.form.Field} field
30185      * @return {Roo.ToolbarItem}
30186      */
30187      
30188       
30189     addField : function(field) {
30190         if (!this.fields) {
30191             var autoId = 0;
30192             this.fields = new Roo.util.MixedCollection(false, function(o){
30193                 return o.id || ("item" + (++autoId));
30194             });
30195
30196         }
30197         
30198         var td = this.nextBlock();
30199         field.render(td);
30200         var ti = new Roo.Toolbar.Item(td.firstChild);
30201         ti.render(td);
30202         this.items.add(ti);
30203         this.fields.add(field);
30204         return ti;
30205     },
30206     /**
30207      * Hide the toolbar
30208      * @method hide
30209      */
30210      
30211       
30212     hide : function()
30213     {
30214         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30215         this.el.child('div').hide();
30216     },
30217     /**
30218      * Show the toolbar
30219      * @method show
30220      */
30221     show : function()
30222     {
30223         this.el.child('div').show();
30224     },
30225       
30226     // private
30227     nextBlock : function(){
30228         var td = document.createElement("td");
30229         this.tr.appendChild(td);
30230         return td;
30231     },
30232
30233     // private
30234     destroy : function(){
30235         if(this.items){ // rendered?
30236             Roo.destroy.apply(Roo, this.items.items);
30237         }
30238         if(this.fields){ // rendered?
30239             Roo.destroy.apply(Roo, this.fields.items);
30240         }
30241         Roo.Element.uncache(this.el, this.tr);
30242     }
30243 };
30244
30245 /**
30246  * @class Roo.Toolbar.Item
30247  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30248  * @constructor
30249  * Creates a new Item
30250  * @param {HTMLElement} el 
30251  */
30252 Roo.Toolbar.Item = function(el){
30253     var cfg = {};
30254     if (typeof (el.xtype) != 'undefined') {
30255         cfg = el;
30256         el = cfg.el;
30257     }
30258     
30259     this.el = Roo.getDom(el);
30260     this.id = Roo.id(this.el);
30261     this.hidden = false;
30262     
30263     this.addEvents({
30264          /**
30265              * @event render
30266              * Fires when the button is rendered
30267              * @param {Button} this
30268              */
30269         'render': true
30270     });
30271     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30272 };
30273 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30274 //Roo.Toolbar.Item.prototype = {
30275     
30276     /**
30277      * Get this item's HTML Element
30278      * @return {HTMLElement}
30279      */
30280     getEl : function(){
30281        return this.el;  
30282     },
30283
30284     // private
30285     render : function(td){
30286         
30287          this.td = td;
30288         td.appendChild(this.el);
30289         
30290         this.fireEvent('render', this);
30291     },
30292     
30293     /**
30294      * Removes and destroys this item.
30295      */
30296     destroy : function(){
30297         this.td.parentNode.removeChild(this.td);
30298     },
30299     
30300     /**
30301      * Shows this item.
30302      */
30303     show: function(){
30304         this.hidden = false;
30305         this.td.style.display = "";
30306     },
30307     
30308     /**
30309      * Hides this item.
30310      */
30311     hide: function(){
30312         this.hidden = true;
30313         this.td.style.display = "none";
30314     },
30315     
30316     /**
30317      * Convenience function for boolean show/hide.
30318      * @param {Boolean} visible true to show/false to hide
30319      */
30320     setVisible: function(visible){
30321         if(visible) {
30322             this.show();
30323         }else{
30324             this.hide();
30325         }
30326     },
30327     
30328     /**
30329      * Try to focus this item.
30330      */
30331     focus : function(){
30332         Roo.fly(this.el).focus();
30333     },
30334     
30335     /**
30336      * Disables this item.
30337      */
30338     disable : function(){
30339         Roo.fly(this.td).addClass("x-item-disabled");
30340         this.disabled = true;
30341         this.el.disabled = true;
30342     },
30343     
30344     /**
30345      * Enables this item.
30346      */
30347     enable : function(){
30348         Roo.fly(this.td).removeClass("x-item-disabled");
30349         this.disabled = false;
30350         this.el.disabled = false;
30351     }
30352 });
30353
30354
30355 /**
30356  * @class Roo.Toolbar.Separator
30357  * @extends Roo.Toolbar.Item
30358  * A simple toolbar separator class
30359  * @constructor
30360  * Creates a new Separator
30361  */
30362 Roo.Toolbar.Separator = function(cfg){
30363     
30364     var s = document.createElement("span");
30365     s.className = "ytb-sep";
30366     if (cfg) {
30367         cfg.el = s;
30368     }
30369     
30370     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30371 };
30372 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30373     enable:Roo.emptyFn,
30374     disable:Roo.emptyFn,
30375     focus:Roo.emptyFn
30376 });
30377
30378 /**
30379  * @class Roo.Toolbar.Spacer
30380  * @extends Roo.Toolbar.Item
30381  * A simple element that adds extra horizontal space to a toolbar.
30382  * @constructor
30383  * Creates a new Spacer
30384  */
30385 Roo.Toolbar.Spacer = function(cfg){
30386     var s = document.createElement("div");
30387     s.className = "ytb-spacer";
30388     if (cfg) {
30389         cfg.el = s;
30390     }
30391     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30392 };
30393 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30394     enable:Roo.emptyFn,
30395     disable:Roo.emptyFn,
30396     focus:Roo.emptyFn
30397 });
30398
30399 /**
30400  * @class Roo.Toolbar.Fill
30401  * @extends Roo.Toolbar.Spacer
30402  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30403  * @constructor
30404  * Creates a new Spacer
30405  */
30406 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30407     // private
30408     render : function(td){
30409         td.style.width = '100%';
30410         Roo.Toolbar.Fill.superclass.render.call(this, td);
30411     }
30412 });
30413
30414 /**
30415  * @class Roo.Toolbar.TextItem
30416  * @extends Roo.Toolbar.Item
30417  * A simple class that renders text directly into a toolbar.
30418  * @constructor
30419  * Creates a new TextItem
30420  * @param {String} text
30421  */
30422 Roo.Toolbar.TextItem = function(cfg){
30423     var  text = cfg || "";
30424     if (typeof(cfg) == 'object') {
30425         text = cfg.text || "";
30426     }  else {
30427         cfg = null;
30428     }
30429     var s = document.createElement("span");
30430     s.className = "ytb-text";
30431     s.innerHTML = text;
30432     if (cfg) {
30433         cfg.el  = s;
30434     }
30435     
30436     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30437 };
30438 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30439     
30440      
30441     enable:Roo.emptyFn,
30442     disable:Roo.emptyFn,
30443     focus:Roo.emptyFn
30444 });
30445
30446 /**
30447  * @class Roo.Toolbar.Button
30448  * @extends Roo.Button
30449  * A button that renders into a toolbar.
30450  * @constructor
30451  * Creates a new Button
30452  * @param {Object} config A standard {@link Roo.Button} config object
30453  */
30454 Roo.Toolbar.Button = function(config){
30455     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30456 };
30457 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30458     render : function(td){
30459         this.td = td;
30460         Roo.Toolbar.Button.superclass.render.call(this, td);
30461     },
30462     
30463     /**
30464      * Removes and destroys this button
30465      */
30466     destroy : function(){
30467         Roo.Toolbar.Button.superclass.destroy.call(this);
30468         this.td.parentNode.removeChild(this.td);
30469     },
30470     
30471     /**
30472      * Shows this button
30473      */
30474     show: function(){
30475         this.hidden = false;
30476         this.td.style.display = "";
30477     },
30478     
30479     /**
30480      * Hides this button
30481      */
30482     hide: function(){
30483         this.hidden = true;
30484         this.td.style.display = "none";
30485     },
30486
30487     /**
30488      * Disables this item
30489      */
30490     disable : function(){
30491         Roo.fly(this.td).addClass("x-item-disabled");
30492         this.disabled = true;
30493     },
30494
30495     /**
30496      * Enables this item
30497      */
30498     enable : function(){
30499         Roo.fly(this.td).removeClass("x-item-disabled");
30500         this.disabled = false;
30501     }
30502 });
30503 // backwards compat
30504 Roo.ToolbarButton = Roo.Toolbar.Button;
30505
30506 /**
30507  * @class Roo.Toolbar.SplitButton
30508  * @extends Roo.SplitButton
30509  * A menu button that renders into a toolbar.
30510  * @constructor
30511  * Creates a new SplitButton
30512  * @param {Object} config A standard {@link Roo.SplitButton} config object
30513  */
30514 Roo.Toolbar.SplitButton = function(config){
30515     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30516 };
30517 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30518     render : function(td){
30519         this.td = td;
30520         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30521     },
30522     
30523     /**
30524      * Removes and destroys this button
30525      */
30526     destroy : function(){
30527         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30528         this.td.parentNode.removeChild(this.td);
30529     },
30530     
30531     /**
30532      * Shows this button
30533      */
30534     show: function(){
30535         this.hidden = false;
30536         this.td.style.display = "";
30537     },
30538     
30539     /**
30540      * Hides this button
30541      */
30542     hide: function(){
30543         this.hidden = true;
30544         this.td.style.display = "none";
30545     }
30546 });
30547
30548 // backwards compat
30549 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30550  * Based on:
30551  * Ext JS Library 1.1.1
30552  * Copyright(c) 2006-2007, Ext JS, LLC.
30553  *
30554  * Originally Released Under LGPL - original licence link has changed is not relivant.
30555  *
30556  * Fork - LGPL
30557  * <script type="text/javascript">
30558  */
30559  
30560 /**
30561  * @class Roo.PagingToolbar
30562  * @extends Roo.Toolbar
30563  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30564  * @constructor
30565  * Create a new PagingToolbar
30566  * @param {Object} config The config object
30567  */
30568 Roo.PagingToolbar = function(el, ds, config)
30569 {
30570     // old args format still supported... - xtype is prefered..
30571     if (typeof(el) == 'object' && el.xtype) {
30572         // created from xtype...
30573         config = el;
30574         ds = el.dataSource;
30575         el = config.container;
30576     }
30577     var items = [];
30578     if (config.items) {
30579         items = config.items;
30580         config.items = [];
30581     }
30582     
30583     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30584     this.ds = ds;
30585     this.cursor = 0;
30586     this.renderButtons(this.el);
30587     this.bind(ds);
30588     
30589     // supprot items array.
30590    
30591     Roo.each(items, function(e) {
30592         this.add(Roo.factory(e));
30593     },this);
30594     
30595 };
30596
30597 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30598     /**
30599      * @cfg {Roo.data.Store} dataSource
30600      * The underlying data store providing the paged data
30601      */
30602     /**
30603      * @cfg {String/HTMLElement/Element} container
30604      * container The id or element that will contain the toolbar
30605      */
30606     /**
30607      * @cfg {Boolean} displayInfo
30608      * True to display the displayMsg (defaults to false)
30609      */
30610     /**
30611      * @cfg {Number} pageSize
30612      * The number of records to display per page (defaults to 20)
30613      */
30614     pageSize: 20,
30615     /**
30616      * @cfg {String} displayMsg
30617      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30618      */
30619     displayMsg : 'Displaying {0} - {1} of {2}',
30620     /**
30621      * @cfg {String} emptyMsg
30622      * The message to display when no records are found (defaults to "No data to display")
30623      */
30624     emptyMsg : 'No data to display',
30625     /**
30626      * Customizable piece of the default paging text (defaults to "Page")
30627      * @type String
30628      */
30629     beforePageText : "Page",
30630     /**
30631      * Customizable piece of the default paging text (defaults to "of %0")
30632      * @type String
30633      */
30634     afterPageText : "of {0}",
30635     /**
30636      * Customizable piece of the default paging text (defaults to "First Page")
30637      * @type String
30638      */
30639     firstText : "First Page",
30640     /**
30641      * Customizable piece of the default paging text (defaults to "Previous Page")
30642      * @type String
30643      */
30644     prevText : "Previous Page",
30645     /**
30646      * Customizable piece of the default paging text (defaults to "Next Page")
30647      * @type String
30648      */
30649     nextText : "Next Page",
30650     /**
30651      * Customizable piece of the default paging text (defaults to "Last Page")
30652      * @type String
30653      */
30654     lastText : "Last Page",
30655     /**
30656      * Customizable piece of the default paging text (defaults to "Refresh")
30657      * @type String
30658      */
30659     refreshText : "Refresh",
30660
30661     // private
30662     renderButtons : function(el){
30663         Roo.PagingToolbar.superclass.render.call(this, el);
30664         this.first = this.addButton({
30665             tooltip: this.firstText,
30666             cls: "x-btn-icon x-grid-page-first",
30667             disabled: true,
30668             handler: this.onClick.createDelegate(this, ["first"])
30669         });
30670         this.prev = this.addButton({
30671             tooltip: this.prevText,
30672             cls: "x-btn-icon x-grid-page-prev",
30673             disabled: true,
30674             handler: this.onClick.createDelegate(this, ["prev"])
30675         });
30676         //this.addSeparator();
30677         this.add(this.beforePageText);
30678         this.field = Roo.get(this.addDom({
30679            tag: "input",
30680            type: "text",
30681            size: "3",
30682            value: "1",
30683            cls: "x-grid-page-number"
30684         }).el);
30685         this.field.on("keydown", this.onPagingKeydown, this);
30686         this.field.on("focus", function(){this.dom.select();});
30687         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30688         this.field.setHeight(18);
30689         //this.addSeparator();
30690         this.next = this.addButton({
30691             tooltip: this.nextText,
30692             cls: "x-btn-icon x-grid-page-next",
30693             disabled: true,
30694             handler: this.onClick.createDelegate(this, ["next"])
30695         });
30696         this.last = this.addButton({
30697             tooltip: this.lastText,
30698             cls: "x-btn-icon x-grid-page-last",
30699             disabled: true,
30700             handler: this.onClick.createDelegate(this, ["last"])
30701         });
30702         //this.addSeparator();
30703         this.loading = this.addButton({
30704             tooltip: this.refreshText,
30705             cls: "x-btn-icon x-grid-loading",
30706             handler: this.onClick.createDelegate(this, ["refresh"])
30707         });
30708
30709         if(this.displayInfo){
30710             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30711         }
30712     },
30713
30714     // private
30715     updateInfo : function(){
30716         if(this.displayEl){
30717             var count = this.ds.getCount();
30718             var msg = count == 0 ?
30719                 this.emptyMsg :
30720                 String.format(
30721                     this.displayMsg,
30722                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30723                 );
30724             this.displayEl.update(msg);
30725         }
30726     },
30727
30728     // private
30729     onLoad : function(ds, r, o){
30730        this.cursor = o.params ? o.params.start : 0;
30731        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30732
30733        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30734        this.field.dom.value = ap;
30735        this.first.setDisabled(ap == 1);
30736        this.prev.setDisabled(ap == 1);
30737        this.next.setDisabled(ap == ps);
30738        this.last.setDisabled(ap == ps);
30739        this.loading.enable();
30740        this.updateInfo();
30741     },
30742
30743     // private
30744     getPageData : function(){
30745         var total = this.ds.getTotalCount();
30746         return {
30747             total : total,
30748             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30749             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30750         };
30751     },
30752
30753     // private
30754     onLoadError : function(){
30755         this.loading.enable();
30756     },
30757
30758     // private
30759     onPagingKeydown : function(e){
30760         var k = e.getKey();
30761         var d = this.getPageData();
30762         if(k == e.RETURN){
30763             var v = this.field.dom.value, pageNum;
30764             if(!v || isNaN(pageNum = parseInt(v, 10))){
30765                 this.field.dom.value = d.activePage;
30766                 return;
30767             }
30768             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30769             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30770             e.stopEvent();
30771         }
30772         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))
30773         {
30774           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30775           this.field.dom.value = pageNum;
30776           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30777           e.stopEvent();
30778         }
30779         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30780         {
30781           var v = this.field.dom.value, pageNum; 
30782           var increment = (e.shiftKey) ? 10 : 1;
30783           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30784             increment *= -1;
30785           }
30786           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30787             this.field.dom.value = d.activePage;
30788             return;
30789           }
30790           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30791           {
30792             this.field.dom.value = parseInt(v, 10) + increment;
30793             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30794             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30795           }
30796           e.stopEvent();
30797         }
30798     },
30799
30800     // private
30801     beforeLoad : function(){
30802         if(this.loading){
30803             this.loading.disable();
30804         }
30805     },
30806
30807     // private
30808     onClick : function(which){
30809         var ds = this.ds;
30810         switch(which){
30811             case "first":
30812                 ds.load({params:{start: 0, limit: this.pageSize}});
30813             break;
30814             case "prev":
30815                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30816             break;
30817             case "next":
30818                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30819             break;
30820             case "last":
30821                 var total = ds.getTotalCount();
30822                 var extra = total % this.pageSize;
30823                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30824                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30825             break;
30826             case "refresh":
30827                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30828             break;
30829         }
30830     },
30831
30832     /**
30833      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30834      * @param {Roo.data.Store} store The data store to unbind
30835      */
30836     unbind : function(ds){
30837         ds.un("beforeload", this.beforeLoad, this);
30838         ds.un("load", this.onLoad, this);
30839         ds.un("loadexception", this.onLoadError, this);
30840         ds.un("remove", this.updateInfo, this);
30841         ds.un("add", this.updateInfo, this);
30842         this.ds = undefined;
30843     },
30844
30845     /**
30846      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30847      * @param {Roo.data.Store} store The data store to bind
30848      */
30849     bind : function(ds){
30850         ds.on("beforeload", this.beforeLoad, this);
30851         ds.on("load", this.onLoad, this);
30852         ds.on("loadexception", this.onLoadError, this);
30853         ds.on("remove", this.updateInfo, this);
30854         ds.on("add", this.updateInfo, this);
30855         this.ds = ds;
30856     }
30857 });/*
30858  * Based on:
30859  * Ext JS Library 1.1.1
30860  * Copyright(c) 2006-2007, Ext JS, LLC.
30861  *
30862  * Originally Released Under LGPL - original licence link has changed is not relivant.
30863  *
30864  * Fork - LGPL
30865  * <script type="text/javascript">
30866  */
30867
30868 /**
30869  * @class Roo.Resizable
30870  * @extends Roo.util.Observable
30871  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30872  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30873  * 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
30874  * the element will be wrapped for you automatically.</p>
30875  * <p>Here is the list of valid resize handles:</p>
30876  * <pre>
30877 Value   Description
30878 ------  -------------------
30879  'n'     north
30880  's'     south
30881  'e'     east
30882  'w'     west
30883  'nw'    northwest
30884  'sw'    southwest
30885  'se'    southeast
30886  'ne'    northeast
30887  'hd'    horizontal drag
30888  'all'   all
30889 </pre>
30890  * <p>Here's an example showing the creation of a typical Resizable:</p>
30891  * <pre><code>
30892 var resizer = new Roo.Resizable("element-id", {
30893     handles: 'all',
30894     minWidth: 200,
30895     minHeight: 100,
30896     maxWidth: 500,
30897     maxHeight: 400,
30898     pinned: true
30899 });
30900 resizer.on("resize", myHandler);
30901 </code></pre>
30902  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30903  * resizer.east.setDisplayed(false);</p>
30904  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30905  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30906  * resize operation's new size (defaults to [0, 0])
30907  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30908  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30909  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30910  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30911  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30912  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30913  * @cfg {Number} width The width of the element in pixels (defaults to null)
30914  * @cfg {Number} height The height of the element in pixels (defaults to null)
30915  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30916  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30917  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30918  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30919  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30920  * in favor of the handles config option (defaults to false)
30921  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30922  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30923  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30924  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30925  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30926  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30927  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30928  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30929  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30930  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30931  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30932  * @constructor
30933  * Create a new resizable component
30934  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30935  * @param {Object} config configuration options
30936   */
30937 Roo.Resizable = function(el, config)
30938 {
30939     this.el = Roo.get(el);
30940
30941     if(config && config.wrap){
30942         config.resizeChild = this.el;
30943         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30944         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30945         this.el.setStyle("overflow", "hidden");
30946         this.el.setPositioning(config.resizeChild.getPositioning());
30947         config.resizeChild.clearPositioning();
30948         if(!config.width || !config.height){
30949             var csize = config.resizeChild.getSize();
30950             this.el.setSize(csize.width, csize.height);
30951         }
30952         if(config.pinned && !config.adjustments){
30953             config.adjustments = "auto";
30954         }
30955     }
30956
30957     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30958     this.proxy.unselectable();
30959     this.proxy.enableDisplayMode('block');
30960
30961     Roo.apply(this, config);
30962
30963     if(this.pinned){
30964         this.disableTrackOver = true;
30965         this.el.addClass("x-resizable-pinned");
30966     }
30967     // if the element isn't positioned, make it relative
30968     var position = this.el.getStyle("position");
30969     if(position != "absolute" && position != "fixed"){
30970         this.el.setStyle("position", "relative");
30971     }
30972     if(!this.handles){ // no handles passed, must be legacy style
30973         this.handles = 's,e,se';
30974         if(this.multiDirectional){
30975             this.handles += ',n,w';
30976         }
30977     }
30978     if(this.handles == "all"){
30979         this.handles = "n s e w ne nw se sw";
30980     }
30981     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30982     var ps = Roo.Resizable.positions;
30983     for(var i = 0, len = hs.length; i < len; i++){
30984         if(hs[i] && ps[hs[i]]){
30985             var pos = ps[hs[i]];
30986             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30987         }
30988     }
30989     // legacy
30990     this.corner = this.southeast;
30991     
30992     // updateBox = the box can move..
30993     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30994         this.updateBox = true;
30995     }
30996
30997     this.activeHandle = null;
30998
30999     if(this.resizeChild){
31000         if(typeof this.resizeChild == "boolean"){
31001             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
31002         }else{
31003             this.resizeChild = Roo.get(this.resizeChild, true);
31004         }
31005     }
31006     
31007     if(this.adjustments == "auto"){
31008         var rc = this.resizeChild;
31009         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31010         if(rc && (hw || hn)){
31011             rc.position("relative");
31012             rc.setLeft(hw ? hw.el.getWidth() : 0);
31013             rc.setTop(hn ? hn.el.getHeight() : 0);
31014         }
31015         this.adjustments = [
31016             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31017             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31018         ];
31019     }
31020
31021     if(this.draggable){
31022         this.dd = this.dynamic ?
31023             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31024         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31025     }
31026
31027     // public events
31028     this.addEvents({
31029         /**
31030          * @event beforeresize
31031          * Fired before resize is allowed. Set enabled to false to cancel resize.
31032          * @param {Roo.Resizable} this
31033          * @param {Roo.EventObject} e The mousedown event
31034          */
31035         "beforeresize" : true,
31036         /**
31037          * @event resizing
31038          * Fired a resizing.
31039          * @param {Roo.Resizable} this
31040          * @param {Number} x The new x position
31041          * @param {Number} y The new y position
31042          * @param {Number} w The new w width
31043          * @param {Number} h The new h hight
31044          * @param {Roo.EventObject} e The mouseup event
31045          */
31046         "resizing" : true,
31047         /**
31048          * @event resize
31049          * Fired after a resize.
31050          * @param {Roo.Resizable} this
31051          * @param {Number} width The new width
31052          * @param {Number} height The new height
31053          * @param {Roo.EventObject} e The mouseup event
31054          */
31055         "resize" : true
31056     });
31057
31058     if(this.width !== null && this.height !== null){
31059         this.resizeTo(this.width, this.height);
31060     }else{
31061         this.updateChildSize();
31062     }
31063     if(Roo.isIE){
31064         this.el.dom.style.zoom = 1;
31065     }
31066     Roo.Resizable.superclass.constructor.call(this);
31067 };
31068
31069 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31070         resizeChild : false,
31071         adjustments : [0, 0],
31072         minWidth : 5,
31073         minHeight : 5,
31074         maxWidth : 10000,
31075         maxHeight : 10000,
31076         enabled : true,
31077         animate : false,
31078         duration : .35,
31079         dynamic : false,
31080         handles : false,
31081         multiDirectional : false,
31082         disableTrackOver : false,
31083         easing : 'easeOutStrong',
31084         widthIncrement : 0,
31085         heightIncrement : 0,
31086         pinned : false,
31087         width : null,
31088         height : null,
31089         preserveRatio : false,
31090         transparent: false,
31091         minX: 0,
31092         minY: 0,
31093         draggable: false,
31094
31095         /**
31096          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31097          */
31098         constrainTo: undefined,
31099         /**
31100          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31101          */
31102         resizeRegion: undefined,
31103
31104
31105     /**
31106      * Perform a manual resize
31107      * @param {Number} width
31108      * @param {Number} height
31109      */
31110     resizeTo : function(width, height){
31111         this.el.setSize(width, height);
31112         this.updateChildSize();
31113         this.fireEvent("resize", this, width, height, null);
31114     },
31115
31116     // private
31117     startSizing : function(e, handle){
31118         this.fireEvent("beforeresize", this, e);
31119         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31120
31121             if(!this.overlay){
31122                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31123                 this.overlay.unselectable();
31124                 this.overlay.enableDisplayMode("block");
31125                 this.overlay.on("mousemove", this.onMouseMove, this);
31126                 this.overlay.on("mouseup", this.onMouseUp, this);
31127             }
31128             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31129
31130             this.resizing = true;
31131             this.startBox = this.el.getBox();
31132             this.startPoint = e.getXY();
31133             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31134                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31135
31136             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31137             this.overlay.show();
31138
31139             if(this.constrainTo) {
31140                 var ct = Roo.get(this.constrainTo);
31141                 this.resizeRegion = ct.getRegion().adjust(
31142                     ct.getFrameWidth('t'),
31143                     ct.getFrameWidth('l'),
31144                     -ct.getFrameWidth('b'),
31145                     -ct.getFrameWidth('r')
31146                 );
31147             }
31148
31149             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31150             this.proxy.show();
31151             this.proxy.setBox(this.startBox);
31152             if(!this.dynamic){
31153                 this.proxy.setStyle('visibility', 'visible');
31154             }
31155         }
31156     },
31157
31158     // private
31159     onMouseDown : function(handle, e){
31160         if(this.enabled){
31161             e.stopEvent();
31162             this.activeHandle = handle;
31163             this.startSizing(e, handle);
31164         }
31165     },
31166
31167     // private
31168     onMouseUp : function(e){
31169         var size = this.resizeElement();
31170         this.resizing = false;
31171         this.handleOut();
31172         this.overlay.hide();
31173         this.proxy.hide();
31174         this.fireEvent("resize", this, size.width, size.height, e);
31175     },
31176
31177     // private
31178     updateChildSize : function(){
31179         
31180         if(this.resizeChild){
31181             var el = this.el;
31182             var child = this.resizeChild;
31183             var adj = this.adjustments;
31184             if(el.dom.offsetWidth){
31185                 var b = el.getSize(true);
31186                 child.setSize(b.width+adj[0], b.height+adj[1]);
31187             }
31188             // Second call here for IE
31189             // The first call enables instant resizing and
31190             // the second call corrects scroll bars if they
31191             // exist
31192             if(Roo.isIE){
31193                 setTimeout(function(){
31194                     if(el.dom.offsetWidth){
31195                         var b = el.getSize(true);
31196                         child.setSize(b.width+adj[0], b.height+adj[1]);
31197                     }
31198                 }, 10);
31199             }
31200         }
31201     },
31202
31203     // private
31204     snap : function(value, inc, min){
31205         if(!inc || !value) {
31206             return value;
31207         }
31208         var newValue = value;
31209         var m = value % inc;
31210         if(m > 0){
31211             if(m > (inc/2)){
31212                 newValue = value + (inc-m);
31213             }else{
31214                 newValue = value - m;
31215             }
31216         }
31217         return Math.max(min, newValue);
31218     },
31219
31220     // private
31221     resizeElement : function(){
31222         var box = this.proxy.getBox();
31223         if(this.updateBox){
31224             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31225         }else{
31226             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31227         }
31228         this.updateChildSize();
31229         if(!this.dynamic){
31230             this.proxy.hide();
31231         }
31232         return box;
31233     },
31234
31235     // private
31236     constrain : function(v, diff, m, mx){
31237         if(v - diff < m){
31238             diff = v - m;
31239         }else if(v - diff > mx){
31240             diff = mx - v;
31241         }
31242         return diff;
31243     },
31244
31245     // private
31246     onMouseMove : function(e){
31247         
31248         if(this.enabled){
31249             try{// try catch so if something goes wrong the user doesn't get hung
31250
31251             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31252                 return;
31253             }
31254
31255             //var curXY = this.startPoint;
31256             var curSize = this.curSize || this.startBox;
31257             var x = this.startBox.x, y = this.startBox.y;
31258             var ox = x, oy = y;
31259             var w = curSize.width, h = curSize.height;
31260             var ow = w, oh = h;
31261             var mw = this.minWidth, mh = this.minHeight;
31262             var mxw = this.maxWidth, mxh = this.maxHeight;
31263             var wi = this.widthIncrement;
31264             var hi = this.heightIncrement;
31265
31266             var eventXY = e.getXY();
31267             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31268             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31269
31270             var pos = this.activeHandle.position;
31271
31272             switch(pos){
31273                 case "east":
31274                     w += diffX;
31275                     w = Math.min(Math.max(mw, w), mxw);
31276                     break;
31277              
31278                 case "south":
31279                     h += diffY;
31280                     h = Math.min(Math.max(mh, h), mxh);
31281                     break;
31282                 case "southeast":
31283                     w += diffX;
31284                     h += diffY;
31285                     w = Math.min(Math.max(mw, w), mxw);
31286                     h = Math.min(Math.max(mh, h), mxh);
31287                     break;
31288                 case "north":
31289                     diffY = this.constrain(h, diffY, mh, mxh);
31290                     y += diffY;
31291                     h -= diffY;
31292                     break;
31293                 case "hdrag":
31294                     
31295                     if (wi) {
31296                         var adiffX = Math.abs(diffX);
31297                         var sub = (adiffX % wi); // how much 
31298                         if (sub > (wi/2)) { // far enough to snap
31299                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31300                         } else {
31301                             // remove difference.. 
31302                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31303                         }
31304                     }
31305                     x += diffX;
31306                     x = Math.max(this.minX, x);
31307                     break;
31308                 case "west":
31309                     diffX = this.constrain(w, diffX, mw, mxw);
31310                     x += diffX;
31311                     w -= diffX;
31312                     break;
31313                 case "northeast":
31314                     w += diffX;
31315                     w = Math.min(Math.max(mw, w), mxw);
31316                     diffY = this.constrain(h, diffY, mh, mxh);
31317                     y += diffY;
31318                     h -= diffY;
31319                     break;
31320                 case "northwest":
31321                     diffX = this.constrain(w, diffX, mw, mxw);
31322                     diffY = this.constrain(h, diffY, mh, mxh);
31323                     y += diffY;
31324                     h -= diffY;
31325                     x += diffX;
31326                     w -= diffX;
31327                     break;
31328                case "southwest":
31329                     diffX = this.constrain(w, diffX, mw, mxw);
31330                     h += diffY;
31331                     h = Math.min(Math.max(mh, h), mxh);
31332                     x += diffX;
31333                     w -= diffX;
31334                     break;
31335             }
31336
31337             var sw = this.snap(w, wi, mw);
31338             var sh = this.snap(h, hi, mh);
31339             if(sw != w || sh != h){
31340                 switch(pos){
31341                     case "northeast":
31342                         y -= sh - h;
31343                     break;
31344                     case "north":
31345                         y -= sh - h;
31346                         break;
31347                     case "southwest":
31348                         x -= sw - w;
31349                     break;
31350                     case "west":
31351                         x -= sw - w;
31352                         break;
31353                     case "northwest":
31354                         x -= sw - w;
31355                         y -= sh - h;
31356                     break;
31357                 }
31358                 w = sw;
31359                 h = sh;
31360             }
31361
31362             if(this.preserveRatio){
31363                 switch(pos){
31364                     case "southeast":
31365                     case "east":
31366                         h = oh * (w/ow);
31367                         h = Math.min(Math.max(mh, h), mxh);
31368                         w = ow * (h/oh);
31369                        break;
31370                     case "south":
31371                         w = ow * (h/oh);
31372                         w = Math.min(Math.max(mw, w), mxw);
31373                         h = oh * (w/ow);
31374                         break;
31375                     case "northeast":
31376                         w = ow * (h/oh);
31377                         w = Math.min(Math.max(mw, w), mxw);
31378                         h = oh * (w/ow);
31379                     break;
31380                     case "north":
31381                         var tw = w;
31382                         w = ow * (h/oh);
31383                         w = Math.min(Math.max(mw, w), mxw);
31384                         h = oh * (w/ow);
31385                         x += (tw - w) / 2;
31386                         break;
31387                     case "southwest":
31388                         h = oh * (w/ow);
31389                         h = Math.min(Math.max(mh, h), mxh);
31390                         var tw = w;
31391                         w = ow * (h/oh);
31392                         x += tw - w;
31393                         break;
31394                     case "west":
31395                         var th = h;
31396                         h = oh * (w/ow);
31397                         h = Math.min(Math.max(mh, h), mxh);
31398                         y += (th - h) / 2;
31399                         var tw = w;
31400                         w = ow * (h/oh);
31401                         x += tw - w;
31402                        break;
31403                     case "northwest":
31404                         var tw = w;
31405                         var th = h;
31406                         h = oh * (w/ow);
31407                         h = Math.min(Math.max(mh, h), mxh);
31408                         w = ow * (h/oh);
31409                         y += th - h;
31410                         x += tw - w;
31411                        break;
31412
31413                 }
31414             }
31415             if (pos == 'hdrag') {
31416                 w = ow;
31417             }
31418             this.proxy.setBounds(x, y, w, h);
31419             if(this.dynamic){
31420                 this.resizeElement();
31421             }
31422             }catch(e){}
31423         }
31424         this.fireEvent("resizing", this, x, y, w, h, e);
31425     },
31426
31427     // private
31428     handleOver : function(){
31429         if(this.enabled){
31430             this.el.addClass("x-resizable-over");
31431         }
31432     },
31433
31434     // private
31435     handleOut : function(){
31436         if(!this.resizing){
31437             this.el.removeClass("x-resizable-over");
31438         }
31439     },
31440
31441     /**
31442      * Returns the element this component is bound to.
31443      * @return {Roo.Element}
31444      */
31445     getEl : function(){
31446         return this.el;
31447     },
31448
31449     /**
31450      * Returns the resizeChild element (or null).
31451      * @return {Roo.Element}
31452      */
31453     getResizeChild : function(){
31454         return this.resizeChild;
31455     },
31456     groupHandler : function()
31457     {
31458         
31459     },
31460     /**
31461      * Destroys this resizable. If the element was wrapped and
31462      * removeEl is not true then the element remains.
31463      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31464      */
31465     destroy : function(removeEl){
31466         this.proxy.remove();
31467         if(this.overlay){
31468             this.overlay.removeAllListeners();
31469             this.overlay.remove();
31470         }
31471         var ps = Roo.Resizable.positions;
31472         for(var k in ps){
31473             if(typeof ps[k] != "function" && this[ps[k]]){
31474                 var h = this[ps[k]];
31475                 h.el.removeAllListeners();
31476                 h.el.remove();
31477             }
31478         }
31479         if(removeEl){
31480             this.el.update("");
31481             this.el.remove();
31482         }
31483     }
31484 });
31485
31486 // private
31487 // hash to map config positions to true positions
31488 Roo.Resizable.positions = {
31489     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31490     hd: "hdrag"
31491 };
31492
31493 // private
31494 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31495     if(!this.tpl){
31496         // only initialize the template if resizable is used
31497         var tpl = Roo.DomHelper.createTemplate(
31498             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31499         );
31500         tpl.compile();
31501         Roo.Resizable.Handle.prototype.tpl = tpl;
31502     }
31503     this.position = pos;
31504     this.rz = rz;
31505     // show north drag fro topdra
31506     var handlepos = pos == 'hdrag' ? 'north' : pos;
31507     
31508     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31509     if (pos == 'hdrag') {
31510         this.el.setStyle('cursor', 'pointer');
31511     }
31512     this.el.unselectable();
31513     if(transparent){
31514         this.el.setOpacity(0);
31515     }
31516     this.el.on("mousedown", this.onMouseDown, this);
31517     if(!disableTrackOver){
31518         this.el.on("mouseover", this.onMouseOver, this);
31519         this.el.on("mouseout", this.onMouseOut, this);
31520     }
31521 };
31522
31523 // private
31524 Roo.Resizable.Handle.prototype = {
31525     afterResize : function(rz){
31526         Roo.log('after?');
31527         // do nothing
31528     },
31529     // private
31530     onMouseDown : function(e){
31531         this.rz.onMouseDown(this, e);
31532     },
31533     // private
31534     onMouseOver : function(e){
31535         this.rz.handleOver(this, e);
31536     },
31537     // private
31538     onMouseOut : function(e){
31539         this.rz.handleOut(this, e);
31540     }
31541 };/*
31542  * Based on:
31543  * Ext JS Library 1.1.1
31544  * Copyright(c) 2006-2007, Ext JS, LLC.
31545  *
31546  * Originally Released Under LGPL - original licence link has changed is not relivant.
31547  *
31548  * Fork - LGPL
31549  * <script type="text/javascript">
31550  */
31551
31552 /**
31553  * @class Roo.Editor
31554  * @extends Roo.Component
31555  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31556  * @constructor
31557  * Create a new Editor
31558  * @param {Roo.form.Field} field The Field object (or descendant)
31559  * @param {Object} config The config object
31560  */
31561 Roo.Editor = function(field, config){
31562     Roo.Editor.superclass.constructor.call(this, config);
31563     this.field = field;
31564     this.addEvents({
31565         /**
31566              * @event beforestartedit
31567              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31568              * false from the handler of this event.
31569              * @param {Editor} this
31570              * @param {Roo.Element} boundEl The underlying element bound to this editor
31571              * @param {Mixed} value The field value being set
31572              */
31573         "beforestartedit" : true,
31574         /**
31575              * @event startedit
31576              * Fires when this editor is displayed
31577              * @param {Roo.Element} boundEl The underlying element bound to this editor
31578              * @param {Mixed} value The starting field value
31579              */
31580         "startedit" : true,
31581         /**
31582              * @event beforecomplete
31583              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31584              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31585              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31586              * event will not fire since no edit actually occurred.
31587              * @param {Editor} this
31588              * @param {Mixed} value The current field value
31589              * @param {Mixed} startValue The original field value
31590              */
31591         "beforecomplete" : true,
31592         /**
31593              * @event complete
31594              * Fires after editing is complete and any changed value has been written to the underlying field.
31595              * @param {Editor} this
31596              * @param {Mixed} value The current field value
31597              * @param {Mixed} startValue The original field value
31598              */
31599         "complete" : true,
31600         /**
31601          * @event specialkey
31602          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31603          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31604          * @param {Roo.form.Field} this
31605          * @param {Roo.EventObject} e The event object
31606          */
31607         "specialkey" : true
31608     });
31609 };
31610
31611 Roo.extend(Roo.Editor, Roo.Component, {
31612     /**
31613      * @cfg {Boolean/String} autosize
31614      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31615      * or "height" to adopt the height only (defaults to false)
31616      */
31617     /**
31618      * @cfg {Boolean} revertInvalid
31619      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31620      * validation fails (defaults to true)
31621      */
31622     /**
31623      * @cfg {Boolean} ignoreNoChange
31624      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31625      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31626      * will never be ignored.
31627      */
31628     /**
31629      * @cfg {Boolean} hideEl
31630      * False to keep the bound element visible while the editor is displayed (defaults to true)
31631      */
31632     /**
31633      * @cfg {Mixed} value
31634      * The data value of the underlying field (defaults to "")
31635      */
31636     value : "",
31637     /**
31638      * @cfg {String} alignment
31639      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31640      */
31641     alignment: "c-c?",
31642     /**
31643      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31644      * for bottom-right shadow (defaults to "frame")
31645      */
31646     shadow : "frame",
31647     /**
31648      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31649      */
31650     constrain : false,
31651     /**
31652      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31653      */
31654     completeOnEnter : false,
31655     /**
31656      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31657      */
31658     cancelOnEsc : false,
31659     /**
31660      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31661      */
31662     updateEl : false,
31663
31664     // private
31665     onRender : function(ct, position){
31666         this.el = new Roo.Layer({
31667             shadow: this.shadow,
31668             cls: "x-editor",
31669             parentEl : ct,
31670             shim : this.shim,
31671             shadowOffset:4,
31672             id: this.id,
31673             constrain: this.constrain
31674         });
31675         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31676         if(this.field.msgTarget != 'title'){
31677             this.field.msgTarget = 'qtip';
31678         }
31679         this.field.render(this.el);
31680         if(Roo.isGecko){
31681             this.field.el.dom.setAttribute('autocomplete', 'off');
31682         }
31683         this.field.on("specialkey", this.onSpecialKey, this);
31684         if(this.swallowKeys){
31685             this.field.el.swallowEvent(['keydown','keypress']);
31686         }
31687         this.field.show();
31688         this.field.on("blur", this.onBlur, this);
31689         if(this.field.grow){
31690             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31691         }
31692     },
31693
31694     onSpecialKey : function(field, e)
31695     {
31696         //Roo.log('editor onSpecialKey');
31697         if(this.completeOnEnter && e.getKey() == e.ENTER){
31698             e.stopEvent();
31699             this.completeEdit();
31700             return;
31701         }
31702         // do not fire special key otherwise it might hide close the editor...
31703         if(e.getKey() == e.ENTER){    
31704             return;
31705         }
31706         if(this.cancelOnEsc && e.getKey() == e.ESC){
31707             this.cancelEdit();
31708             return;
31709         } 
31710         this.fireEvent('specialkey', field, e);
31711     
31712     },
31713
31714     /**
31715      * Starts the editing process and shows the editor.
31716      * @param {String/HTMLElement/Element} el The element to edit
31717      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31718       * to the innerHTML of el.
31719      */
31720     startEdit : function(el, value){
31721         if(this.editing){
31722             this.completeEdit();
31723         }
31724         this.boundEl = Roo.get(el);
31725         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31726         if(!this.rendered){
31727             this.render(this.parentEl || document.body);
31728         }
31729         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31730             return;
31731         }
31732         this.startValue = v;
31733         this.field.setValue(v);
31734         if(this.autoSize){
31735             var sz = this.boundEl.getSize();
31736             switch(this.autoSize){
31737                 case "width":
31738                 this.setSize(sz.width,  "");
31739                 break;
31740                 case "height":
31741                 this.setSize("",  sz.height);
31742                 break;
31743                 default:
31744                 this.setSize(sz.width,  sz.height);
31745             }
31746         }
31747         this.el.alignTo(this.boundEl, this.alignment);
31748         this.editing = true;
31749         if(Roo.QuickTips){
31750             Roo.QuickTips.disable();
31751         }
31752         this.show();
31753     },
31754
31755     /**
31756      * Sets the height and width of this editor.
31757      * @param {Number} width The new width
31758      * @param {Number} height The new height
31759      */
31760     setSize : function(w, h){
31761         this.field.setSize(w, h);
31762         if(this.el){
31763             this.el.sync();
31764         }
31765     },
31766
31767     /**
31768      * Realigns the editor to the bound field based on the current alignment config value.
31769      */
31770     realign : function(){
31771         this.el.alignTo(this.boundEl, this.alignment);
31772     },
31773
31774     /**
31775      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31776      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31777      */
31778     completeEdit : function(remainVisible){
31779         if(!this.editing){
31780             return;
31781         }
31782         var v = this.getValue();
31783         if(this.revertInvalid !== false && !this.field.isValid()){
31784             v = this.startValue;
31785             this.cancelEdit(true);
31786         }
31787         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31788             this.editing = false;
31789             this.hide();
31790             return;
31791         }
31792         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31793             this.editing = false;
31794             if(this.updateEl && this.boundEl){
31795                 this.boundEl.update(v);
31796             }
31797             if(remainVisible !== true){
31798                 this.hide();
31799             }
31800             this.fireEvent("complete", this, v, this.startValue);
31801         }
31802     },
31803
31804     // private
31805     onShow : function(){
31806         this.el.show();
31807         if(this.hideEl !== false){
31808             this.boundEl.hide();
31809         }
31810         this.field.show();
31811         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31812             this.fixIEFocus = true;
31813             this.deferredFocus.defer(50, this);
31814         }else{
31815             this.field.focus();
31816         }
31817         this.fireEvent("startedit", this.boundEl, this.startValue);
31818     },
31819
31820     deferredFocus : function(){
31821         if(this.editing){
31822             this.field.focus();
31823         }
31824     },
31825
31826     /**
31827      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31828      * reverted to the original starting value.
31829      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31830      * cancel (defaults to false)
31831      */
31832     cancelEdit : function(remainVisible){
31833         if(this.editing){
31834             this.setValue(this.startValue);
31835             if(remainVisible !== true){
31836                 this.hide();
31837             }
31838         }
31839     },
31840
31841     // private
31842     onBlur : function(){
31843         if(this.allowBlur !== true && this.editing){
31844             this.completeEdit();
31845         }
31846     },
31847
31848     // private
31849     onHide : function(){
31850         if(this.editing){
31851             this.completeEdit();
31852             return;
31853         }
31854         this.field.blur();
31855         if(this.field.collapse){
31856             this.field.collapse();
31857         }
31858         this.el.hide();
31859         if(this.hideEl !== false){
31860             this.boundEl.show();
31861         }
31862         if(Roo.QuickTips){
31863             Roo.QuickTips.enable();
31864         }
31865     },
31866
31867     /**
31868      * Sets the data value of the editor
31869      * @param {Mixed} value Any valid value supported by the underlying field
31870      */
31871     setValue : function(v){
31872         this.field.setValue(v);
31873     },
31874
31875     /**
31876      * Gets the data value of the editor
31877      * @return {Mixed} The data value
31878      */
31879     getValue : function(){
31880         return this.field.getValue();
31881     }
31882 });/*
31883  * Based on:
31884  * Ext JS Library 1.1.1
31885  * Copyright(c) 2006-2007, Ext JS, LLC.
31886  *
31887  * Originally Released Under LGPL - original licence link has changed is not relivant.
31888  *
31889  * Fork - LGPL
31890  * <script type="text/javascript">
31891  */
31892  
31893 /**
31894  * @class Roo.BasicDialog
31895  * @extends Roo.util.Observable
31896  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31897  * <pre><code>
31898 var dlg = new Roo.BasicDialog("my-dlg", {
31899     height: 200,
31900     width: 300,
31901     minHeight: 100,
31902     minWidth: 150,
31903     modal: true,
31904     proxyDrag: true,
31905     shadow: true
31906 });
31907 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31908 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31909 dlg.addButton('Cancel', dlg.hide, dlg);
31910 dlg.show();
31911 </code></pre>
31912   <b>A Dialog should always be a direct child of the body element.</b>
31913  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31914  * @cfg {String} title Default text to display in the title bar (defaults to null)
31915  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31916  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31917  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31918  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31919  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31920  * (defaults to null with no animation)
31921  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31922  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31923  * property for valid values (defaults to 'all')
31924  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31925  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31926  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31927  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31928  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31929  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31930  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31931  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31932  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31933  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31934  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31935  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31936  * draggable = true (defaults to false)
31937  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31938  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31939  * shadow (defaults to false)
31940  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31941  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31942  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31943  * @cfg {Array} buttons Array of buttons
31944  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31945  * @constructor
31946  * Create a new BasicDialog.
31947  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31948  * @param {Object} config Configuration options
31949  */
31950 Roo.BasicDialog = function(el, config){
31951     this.el = Roo.get(el);
31952     var dh = Roo.DomHelper;
31953     if(!this.el && config && config.autoCreate){
31954         if(typeof config.autoCreate == "object"){
31955             if(!config.autoCreate.id){
31956                 config.autoCreate.id = el;
31957             }
31958             this.el = dh.append(document.body,
31959                         config.autoCreate, true);
31960         }else{
31961             this.el = dh.append(document.body,
31962                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31963         }
31964     }
31965     el = this.el;
31966     el.setDisplayed(true);
31967     el.hide = this.hideAction;
31968     this.id = el.id;
31969     el.addClass("x-dlg");
31970
31971     Roo.apply(this, config);
31972
31973     this.proxy = el.createProxy("x-dlg-proxy");
31974     this.proxy.hide = this.hideAction;
31975     this.proxy.setOpacity(.5);
31976     this.proxy.hide();
31977
31978     if(config.width){
31979         el.setWidth(config.width);
31980     }
31981     if(config.height){
31982         el.setHeight(config.height);
31983     }
31984     this.size = el.getSize();
31985     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31986         this.xy = [config.x,config.y];
31987     }else{
31988         this.xy = el.getCenterXY(true);
31989     }
31990     /** The header element @type Roo.Element */
31991     this.header = el.child("> .x-dlg-hd");
31992     /** The body element @type Roo.Element */
31993     this.body = el.child("> .x-dlg-bd");
31994     /** The footer element @type Roo.Element */
31995     this.footer = el.child("> .x-dlg-ft");
31996
31997     if(!this.header){
31998         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31999     }
32000     if(!this.body){
32001         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
32002     }
32003
32004     this.header.unselectable();
32005     if(this.title){
32006         this.header.update(this.title);
32007     }
32008     // this element allows the dialog to be focused for keyboard event
32009     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32010     this.focusEl.swallowEvent("click", true);
32011
32012     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32013
32014     // wrap the body and footer for special rendering
32015     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32016     if(this.footer){
32017         this.bwrap.dom.appendChild(this.footer.dom);
32018     }
32019
32020     this.bg = this.el.createChild({
32021         tag: "div", cls:"x-dlg-bg",
32022         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32023     });
32024     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32025
32026
32027     if(this.autoScroll !== false && !this.autoTabs){
32028         this.body.setStyle("overflow", "auto");
32029     }
32030
32031     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32032
32033     if(this.closable !== false){
32034         this.el.addClass("x-dlg-closable");
32035         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32036         this.close.on("click", this.closeClick, this);
32037         this.close.addClassOnOver("x-dlg-close-over");
32038     }
32039     if(this.collapsible !== false){
32040         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32041         this.collapseBtn.on("click", this.collapseClick, this);
32042         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32043         this.header.on("dblclick", this.collapseClick, this);
32044     }
32045     if(this.resizable !== false){
32046         this.el.addClass("x-dlg-resizable");
32047         this.resizer = new Roo.Resizable(el, {
32048             minWidth: this.minWidth || 80,
32049             minHeight:this.minHeight || 80,
32050             handles: this.resizeHandles || "all",
32051             pinned: true
32052         });
32053         this.resizer.on("beforeresize", this.beforeResize, this);
32054         this.resizer.on("resize", this.onResize, this);
32055     }
32056     if(this.draggable !== false){
32057         el.addClass("x-dlg-draggable");
32058         if (!this.proxyDrag) {
32059             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32060         }
32061         else {
32062             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32063         }
32064         dd.setHandleElId(this.header.id);
32065         dd.endDrag = this.endMove.createDelegate(this);
32066         dd.startDrag = this.startMove.createDelegate(this);
32067         dd.onDrag = this.onDrag.createDelegate(this);
32068         dd.scroll = false;
32069         this.dd = dd;
32070     }
32071     if(this.modal){
32072         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32073         this.mask.enableDisplayMode("block");
32074         this.mask.hide();
32075         this.el.addClass("x-dlg-modal");
32076     }
32077     if(this.shadow){
32078         this.shadow = new Roo.Shadow({
32079             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32080             offset : this.shadowOffset
32081         });
32082     }else{
32083         this.shadowOffset = 0;
32084     }
32085     if(Roo.useShims && this.shim !== false){
32086         this.shim = this.el.createShim();
32087         this.shim.hide = this.hideAction;
32088         this.shim.hide();
32089     }else{
32090         this.shim = false;
32091     }
32092     if(this.autoTabs){
32093         this.initTabs();
32094     }
32095     if (this.buttons) { 
32096         var bts= this.buttons;
32097         this.buttons = [];
32098         Roo.each(bts, function(b) {
32099             this.addButton(b);
32100         }, this);
32101     }
32102     
32103     
32104     this.addEvents({
32105         /**
32106          * @event keydown
32107          * Fires when a key is pressed
32108          * @param {Roo.BasicDialog} this
32109          * @param {Roo.EventObject} e
32110          */
32111         "keydown" : true,
32112         /**
32113          * @event move
32114          * Fires when this dialog is moved by the user.
32115          * @param {Roo.BasicDialog} this
32116          * @param {Number} x The new page X
32117          * @param {Number} y The new page Y
32118          */
32119         "move" : true,
32120         /**
32121          * @event resize
32122          * Fires when this dialog is resized by the user.
32123          * @param {Roo.BasicDialog} this
32124          * @param {Number} width The new width
32125          * @param {Number} height The new height
32126          */
32127         "resize" : true,
32128         /**
32129          * @event beforehide
32130          * Fires before this dialog is hidden.
32131          * @param {Roo.BasicDialog} this
32132          */
32133         "beforehide" : true,
32134         /**
32135          * @event hide
32136          * Fires when this dialog is hidden.
32137          * @param {Roo.BasicDialog} this
32138          */
32139         "hide" : true,
32140         /**
32141          * @event beforeshow
32142          * Fires before this dialog is shown.
32143          * @param {Roo.BasicDialog} this
32144          */
32145         "beforeshow" : true,
32146         /**
32147          * @event show
32148          * Fires when this dialog is shown.
32149          * @param {Roo.BasicDialog} this
32150          */
32151         "show" : true
32152     });
32153     el.on("keydown", this.onKeyDown, this);
32154     el.on("mousedown", this.toFront, this);
32155     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32156     this.el.hide();
32157     Roo.DialogManager.register(this);
32158     Roo.BasicDialog.superclass.constructor.call(this);
32159 };
32160
32161 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32162     shadowOffset: Roo.isIE ? 6 : 5,
32163     minHeight: 80,
32164     minWidth: 200,
32165     minButtonWidth: 75,
32166     defaultButton: null,
32167     buttonAlign: "right",
32168     tabTag: 'div',
32169     firstShow: true,
32170
32171     /**
32172      * Sets the dialog title text
32173      * @param {String} text The title text to display
32174      * @return {Roo.BasicDialog} this
32175      */
32176     setTitle : function(text){
32177         this.header.update(text);
32178         return this;
32179     },
32180
32181     // private
32182     closeClick : function(){
32183         this.hide();
32184     },
32185
32186     // private
32187     collapseClick : function(){
32188         this[this.collapsed ? "expand" : "collapse"]();
32189     },
32190
32191     /**
32192      * Collapses the dialog to its minimized state (only the title bar is visible).
32193      * Equivalent to the user clicking the collapse dialog button.
32194      */
32195     collapse : function(){
32196         if(!this.collapsed){
32197             this.collapsed = true;
32198             this.el.addClass("x-dlg-collapsed");
32199             this.restoreHeight = this.el.getHeight();
32200             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32201         }
32202     },
32203
32204     /**
32205      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32206      * clicking the expand dialog button.
32207      */
32208     expand : function(){
32209         if(this.collapsed){
32210             this.collapsed = false;
32211             this.el.removeClass("x-dlg-collapsed");
32212             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32213         }
32214     },
32215
32216     /**
32217      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32218      * @return {Roo.TabPanel} The tabs component
32219      */
32220     initTabs : function(){
32221         var tabs = this.getTabs();
32222         while(tabs.getTab(0)){
32223             tabs.removeTab(0);
32224         }
32225         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32226             var dom = el.dom;
32227             tabs.addTab(Roo.id(dom), dom.title);
32228             dom.title = "";
32229         });
32230         tabs.activate(0);
32231         return tabs;
32232     },
32233
32234     // private
32235     beforeResize : function(){
32236         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32237     },
32238
32239     // private
32240     onResize : function(){
32241         this.refreshSize();
32242         this.syncBodyHeight();
32243         this.adjustAssets();
32244         this.focus();
32245         this.fireEvent("resize", this, this.size.width, this.size.height);
32246     },
32247
32248     // private
32249     onKeyDown : function(e){
32250         if(this.isVisible()){
32251             this.fireEvent("keydown", this, e);
32252         }
32253     },
32254
32255     /**
32256      * Resizes the dialog.
32257      * @param {Number} width
32258      * @param {Number} height
32259      * @return {Roo.BasicDialog} this
32260      */
32261     resizeTo : function(width, height){
32262         this.el.setSize(width, height);
32263         this.size = {width: width, height: height};
32264         this.syncBodyHeight();
32265         if(this.fixedcenter){
32266             this.center();
32267         }
32268         if(this.isVisible()){
32269             this.constrainXY();
32270             this.adjustAssets();
32271         }
32272         this.fireEvent("resize", this, width, height);
32273         return this;
32274     },
32275
32276
32277     /**
32278      * Resizes the dialog to fit the specified content size.
32279      * @param {Number} width
32280      * @param {Number} height
32281      * @return {Roo.BasicDialog} this
32282      */
32283     setContentSize : function(w, h){
32284         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32285         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32286         //if(!this.el.isBorderBox()){
32287             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32288             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32289         //}
32290         if(this.tabs){
32291             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32292             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32293         }
32294         this.resizeTo(w, h);
32295         return this;
32296     },
32297
32298     /**
32299      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32300      * executed in response to a particular key being pressed while the dialog is active.
32301      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32302      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32303      * @param {Function} fn The function to call
32304      * @param {Object} scope (optional) The scope of the function
32305      * @return {Roo.BasicDialog} this
32306      */
32307     addKeyListener : function(key, fn, scope){
32308         var keyCode, shift, ctrl, alt;
32309         if(typeof key == "object" && !(key instanceof Array)){
32310             keyCode = key["key"];
32311             shift = key["shift"];
32312             ctrl = key["ctrl"];
32313             alt = key["alt"];
32314         }else{
32315             keyCode = key;
32316         }
32317         var handler = function(dlg, e){
32318             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32319                 var k = e.getKey();
32320                 if(keyCode instanceof Array){
32321                     for(var i = 0, len = keyCode.length; i < len; i++){
32322                         if(keyCode[i] == k){
32323                           fn.call(scope || window, dlg, k, e);
32324                           return;
32325                         }
32326                     }
32327                 }else{
32328                     if(k == keyCode){
32329                         fn.call(scope || window, dlg, k, e);
32330                     }
32331                 }
32332             }
32333         };
32334         this.on("keydown", handler);
32335         return this;
32336     },
32337
32338     /**
32339      * Returns the TabPanel component (creates it if it doesn't exist).
32340      * Note: If you wish to simply check for the existence of tabs without creating them,
32341      * check for a null 'tabs' property.
32342      * @return {Roo.TabPanel} The tabs component
32343      */
32344     getTabs : function(){
32345         if(!this.tabs){
32346             this.el.addClass("x-dlg-auto-tabs");
32347             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32348             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32349         }
32350         return this.tabs;
32351     },
32352
32353     /**
32354      * Adds a button to the footer section of the dialog.
32355      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32356      * object or a valid Roo.DomHelper element config
32357      * @param {Function} handler The function called when the button is clicked
32358      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32359      * @return {Roo.Button} The new button
32360      */
32361     addButton : function(config, handler, scope){
32362         var dh = Roo.DomHelper;
32363         if(!this.footer){
32364             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32365         }
32366         if(!this.btnContainer){
32367             var tb = this.footer.createChild({
32368
32369                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32370                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32371             }, null, true);
32372             this.btnContainer = tb.firstChild.firstChild.firstChild;
32373         }
32374         var bconfig = {
32375             handler: handler,
32376             scope: scope,
32377             minWidth: this.minButtonWidth,
32378             hideParent:true
32379         };
32380         if(typeof config == "string"){
32381             bconfig.text = config;
32382         }else{
32383             if(config.tag){
32384                 bconfig.dhconfig = config;
32385             }else{
32386                 Roo.apply(bconfig, config);
32387             }
32388         }
32389         var fc = false;
32390         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32391             bconfig.position = Math.max(0, bconfig.position);
32392             fc = this.btnContainer.childNodes[bconfig.position];
32393         }
32394          
32395         var btn = new Roo.Button(
32396             fc ? 
32397                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32398                 : this.btnContainer.appendChild(document.createElement("td")),
32399             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32400             bconfig
32401         );
32402         this.syncBodyHeight();
32403         if(!this.buttons){
32404             /**
32405              * Array of all the buttons that have been added to this dialog via addButton
32406              * @type Array
32407              */
32408             this.buttons = [];
32409         }
32410         this.buttons.push(btn);
32411         return btn;
32412     },
32413
32414     /**
32415      * Sets the default button to be focused when the dialog is displayed.
32416      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32417      * @return {Roo.BasicDialog} this
32418      */
32419     setDefaultButton : function(btn){
32420         this.defaultButton = btn;
32421         return this;
32422     },
32423
32424     // private
32425     getHeaderFooterHeight : function(safe){
32426         var height = 0;
32427         if(this.header){
32428            height += this.header.getHeight();
32429         }
32430         if(this.footer){
32431            var fm = this.footer.getMargins();
32432             height += (this.footer.getHeight()+fm.top+fm.bottom);
32433         }
32434         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32435         height += this.centerBg.getPadding("tb");
32436         return height;
32437     },
32438
32439     // private
32440     syncBodyHeight : function()
32441     {
32442         var bd = this.body, // the text
32443             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32444             bw = this.bwrap;
32445         var height = this.size.height - this.getHeaderFooterHeight(false);
32446         bd.setHeight(height-bd.getMargins("tb"));
32447         var hh = this.header.getHeight();
32448         var h = this.size.height-hh;
32449         cb.setHeight(h);
32450         
32451         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32452         bw.setHeight(h-cb.getPadding("tb"));
32453         
32454         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32455         bd.setWidth(bw.getWidth(true));
32456         if(this.tabs){
32457             this.tabs.syncHeight();
32458             if(Roo.isIE){
32459                 this.tabs.el.repaint();
32460             }
32461         }
32462     },
32463
32464     /**
32465      * Restores the previous state of the dialog if Roo.state is configured.
32466      * @return {Roo.BasicDialog} this
32467      */
32468     restoreState : function(){
32469         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32470         if(box && box.width){
32471             this.xy = [box.x, box.y];
32472             this.resizeTo(box.width, box.height);
32473         }
32474         return this;
32475     },
32476
32477     // private
32478     beforeShow : function(){
32479         this.expand();
32480         if(this.fixedcenter){
32481             this.xy = this.el.getCenterXY(true);
32482         }
32483         if(this.modal){
32484             Roo.get(document.body).addClass("x-body-masked");
32485             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32486             this.mask.show();
32487         }
32488         this.constrainXY();
32489     },
32490
32491     // private
32492     animShow : function(){
32493         var b = Roo.get(this.animateTarget).getBox();
32494         this.proxy.setSize(b.width, b.height);
32495         this.proxy.setLocation(b.x, b.y);
32496         this.proxy.show();
32497         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32498                     true, .35, this.showEl.createDelegate(this));
32499     },
32500
32501     /**
32502      * Shows the dialog.
32503      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32504      * @return {Roo.BasicDialog} this
32505      */
32506     show : function(animateTarget){
32507         if (this.fireEvent("beforeshow", this) === false){
32508             return;
32509         }
32510         if(this.syncHeightBeforeShow){
32511             this.syncBodyHeight();
32512         }else if(this.firstShow){
32513             this.firstShow = false;
32514             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32515         }
32516         this.animateTarget = animateTarget || this.animateTarget;
32517         if(!this.el.isVisible()){
32518             this.beforeShow();
32519             if(this.animateTarget && Roo.get(this.animateTarget)){
32520                 this.animShow();
32521             }else{
32522                 this.showEl();
32523             }
32524         }
32525         return this;
32526     },
32527
32528     // private
32529     showEl : function(){
32530         this.proxy.hide();
32531         this.el.setXY(this.xy);
32532         this.el.show();
32533         this.adjustAssets(true);
32534         this.toFront();
32535         this.focus();
32536         // IE peekaboo bug - fix found by Dave Fenwick
32537         if(Roo.isIE){
32538             this.el.repaint();
32539         }
32540         this.fireEvent("show", this);
32541     },
32542
32543     /**
32544      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32545      * dialog itself will receive focus.
32546      */
32547     focus : function(){
32548         if(this.defaultButton){
32549             this.defaultButton.focus();
32550         }else{
32551             this.focusEl.focus();
32552         }
32553     },
32554
32555     // private
32556     constrainXY : function(){
32557         if(this.constraintoviewport !== false){
32558             if(!this.viewSize){
32559                 if(this.container){
32560                     var s = this.container.getSize();
32561                     this.viewSize = [s.width, s.height];
32562                 }else{
32563                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32564                 }
32565             }
32566             var s = Roo.get(this.container||document).getScroll();
32567
32568             var x = this.xy[0], y = this.xy[1];
32569             var w = this.size.width, h = this.size.height;
32570             var vw = this.viewSize[0], vh = this.viewSize[1];
32571             // only move it if it needs it
32572             var moved = false;
32573             // first validate right/bottom
32574             if(x + w > vw+s.left){
32575                 x = vw - w;
32576                 moved = true;
32577             }
32578             if(y + h > vh+s.top){
32579                 y = vh - h;
32580                 moved = true;
32581             }
32582             // then make sure top/left isn't negative
32583             if(x < s.left){
32584                 x = s.left;
32585                 moved = true;
32586             }
32587             if(y < s.top){
32588                 y = s.top;
32589                 moved = true;
32590             }
32591             if(moved){
32592                 // cache xy
32593                 this.xy = [x, y];
32594                 if(this.isVisible()){
32595                     this.el.setLocation(x, y);
32596                     this.adjustAssets();
32597                 }
32598             }
32599         }
32600     },
32601
32602     // private
32603     onDrag : function(){
32604         if(!this.proxyDrag){
32605             this.xy = this.el.getXY();
32606             this.adjustAssets();
32607         }
32608     },
32609
32610     // private
32611     adjustAssets : function(doShow){
32612         var x = this.xy[0], y = this.xy[1];
32613         var w = this.size.width, h = this.size.height;
32614         if(doShow === true){
32615             if(this.shadow){
32616                 this.shadow.show(this.el);
32617             }
32618             if(this.shim){
32619                 this.shim.show();
32620             }
32621         }
32622         if(this.shadow && this.shadow.isVisible()){
32623             this.shadow.show(this.el);
32624         }
32625         if(this.shim && this.shim.isVisible()){
32626             this.shim.setBounds(x, y, w, h);
32627         }
32628     },
32629
32630     // private
32631     adjustViewport : function(w, h){
32632         if(!w || !h){
32633             w = Roo.lib.Dom.getViewWidth();
32634             h = Roo.lib.Dom.getViewHeight();
32635         }
32636         // cache the size
32637         this.viewSize = [w, h];
32638         if(this.modal && this.mask.isVisible()){
32639             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32640             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32641         }
32642         if(this.isVisible()){
32643             this.constrainXY();
32644         }
32645     },
32646
32647     /**
32648      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32649      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32650      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32651      */
32652     destroy : function(removeEl){
32653         if(this.isVisible()){
32654             this.animateTarget = null;
32655             this.hide();
32656         }
32657         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32658         if(this.tabs){
32659             this.tabs.destroy(removeEl);
32660         }
32661         Roo.destroy(
32662              this.shim,
32663              this.proxy,
32664              this.resizer,
32665              this.close,
32666              this.mask
32667         );
32668         if(this.dd){
32669             this.dd.unreg();
32670         }
32671         if(this.buttons){
32672            for(var i = 0, len = this.buttons.length; i < len; i++){
32673                this.buttons[i].destroy();
32674            }
32675         }
32676         this.el.removeAllListeners();
32677         if(removeEl === true){
32678             this.el.update("");
32679             this.el.remove();
32680         }
32681         Roo.DialogManager.unregister(this);
32682     },
32683
32684     // private
32685     startMove : function(){
32686         if(this.proxyDrag){
32687             this.proxy.show();
32688         }
32689         if(this.constraintoviewport !== false){
32690             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32691         }
32692     },
32693
32694     // private
32695     endMove : function(){
32696         if(!this.proxyDrag){
32697             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32698         }else{
32699             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32700             this.proxy.hide();
32701         }
32702         this.refreshSize();
32703         this.adjustAssets();
32704         this.focus();
32705         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32706     },
32707
32708     /**
32709      * Brings this dialog to the front of any other visible dialogs
32710      * @return {Roo.BasicDialog} this
32711      */
32712     toFront : function(){
32713         Roo.DialogManager.bringToFront(this);
32714         return this;
32715     },
32716
32717     /**
32718      * Sends this dialog to the back (under) of any other visible dialogs
32719      * @return {Roo.BasicDialog} this
32720      */
32721     toBack : function(){
32722         Roo.DialogManager.sendToBack(this);
32723         return this;
32724     },
32725
32726     /**
32727      * Centers this dialog in the viewport
32728      * @return {Roo.BasicDialog} this
32729      */
32730     center : function(){
32731         var xy = this.el.getCenterXY(true);
32732         this.moveTo(xy[0], xy[1]);
32733         return this;
32734     },
32735
32736     /**
32737      * Moves the dialog's top-left corner to the specified point
32738      * @param {Number} x
32739      * @param {Number} y
32740      * @return {Roo.BasicDialog} this
32741      */
32742     moveTo : function(x, y){
32743         this.xy = [x,y];
32744         if(this.isVisible()){
32745             this.el.setXY(this.xy);
32746             this.adjustAssets();
32747         }
32748         return this;
32749     },
32750
32751     /**
32752      * Aligns the dialog to the specified element
32753      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32754      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32755      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32756      * @return {Roo.BasicDialog} this
32757      */
32758     alignTo : function(element, position, offsets){
32759         this.xy = this.el.getAlignToXY(element, position, offsets);
32760         if(this.isVisible()){
32761             this.el.setXY(this.xy);
32762             this.adjustAssets();
32763         }
32764         return this;
32765     },
32766
32767     /**
32768      * Anchors an element to another element and realigns it when the window is resized.
32769      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32770      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32771      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32772      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32773      * is a number, it is used as the buffer delay (defaults to 50ms).
32774      * @return {Roo.BasicDialog} this
32775      */
32776     anchorTo : function(el, alignment, offsets, monitorScroll){
32777         var action = function(){
32778             this.alignTo(el, alignment, offsets);
32779         };
32780         Roo.EventManager.onWindowResize(action, this);
32781         var tm = typeof monitorScroll;
32782         if(tm != 'undefined'){
32783             Roo.EventManager.on(window, 'scroll', action, this,
32784                 {buffer: tm == 'number' ? monitorScroll : 50});
32785         }
32786         action.call(this);
32787         return this;
32788     },
32789
32790     /**
32791      * Returns true if the dialog is visible
32792      * @return {Boolean}
32793      */
32794     isVisible : function(){
32795         return this.el.isVisible();
32796     },
32797
32798     // private
32799     animHide : function(callback){
32800         var b = Roo.get(this.animateTarget).getBox();
32801         this.proxy.show();
32802         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32803         this.el.hide();
32804         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32805                     this.hideEl.createDelegate(this, [callback]));
32806     },
32807
32808     /**
32809      * Hides the dialog.
32810      * @param {Function} callback (optional) Function to call when the dialog is hidden
32811      * @return {Roo.BasicDialog} this
32812      */
32813     hide : function(callback){
32814         if (this.fireEvent("beforehide", this) === false){
32815             return;
32816         }
32817         if(this.shadow){
32818             this.shadow.hide();
32819         }
32820         if(this.shim) {
32821           this.shim.hide();
32822         }
32823         // sometimes animateTarget seems to get set.. causing problems...
32824         // this just double checks..
32825         if(this.animateTarget && Roo.get(this.animateTarget)) {
32826            this.animHide(callback);
32827         }else{
32828             this.el.hide();
32829             this.hideEl(callback);
32830         }
32831         return this;
32832     },
32833
32834     // private
32835     hideEl : function(callback){
32836         this.proxy.hide();
32837         if(this.modal){
32838             this.mask.hide();
32839             Roo.get(document.body).removeClass("x-body-masked");
32840         }
32841         this.fireEvent("hide", this);
32842         if(typeof callback == "function"){
32843             callback();
32844         }
32845     },
32846
32847     // private
32848     hideAction : function(){
32849         this.setLeft("-10000px");
32850         this.setTop("-10000px");
32851         this.setStyle("visibility", "hidden");
32852     },
32853
32854     // private
32855     refreshSize : function(){
32856         this.size = this.el.getSize();
32857         this.xy = this.el.getXY();
32858         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32859     },
32860
32861     // private
32862     // z-index is managed by the DialogManager and may be overwritten at any time
32863     setZIndex : function(index){
32864         if(this.modal){
32865             this.mask.setStyle("z-index", index);
32866         }
32867         if(this.shim){
32868             this.shim.setStyle("z-index", ++index);
32869         }
32870         if(this.shadow){
32871             this.shadow.setZIndex(++index);
32872         }
32873         this.el.setStyle("z-index", ++index);
32874         if(this.proxy){
32875             this.proxy.setStyle("z-index", ++index);
32876         }
32877         if(this.resizer){
32878             this.resizer.proxy.setStyle("z-index", ++index);
32879         }
32880
32881         this.lastZIndex = index;
32882     },
32883
32884     /**
32885      * Returns the element for this dialog
32886      * @return {Roo.Element} The underlying dialog Element
32887      */
32888     getEl : function(){
32889         return this.el;
32890     }
32891 });
32892
32893 /**
32894  * @class Roo.DialogManager
32895  * Provides global access to BasicDialogs that have been created and
32896  * support for z-indexing (layering) multiple open dialogs.
32897  */
32898 Roo.DialogManager = function(){
32899     var list = {};
32900     var accessList = [];
32901     var front = null;
32902
32903     // private
32904     var sortDialogs = function(d1, d2){
32905         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32906     };
32907
32908     // private
32909     var orderDialogs = function(){
32910         accessList.sort(sortDialogs);
32911         var seed = Roo.DialogManager.zseed;
32912         for(var i = 0, len = accessList.length; i < len; i++){
32913             var dlg = accessList[i];
32914             if(dlg){
32915                 dlg.setZIndex(seed + (i*10));
32916             }
32917         }
32918     };
32919
32920     return {
32921         /**
32922          * The starting z-index for BasicDialogs (defaults to 9000)
32923          * @type Number The z-index value
32924          */
32925         zseed : 9000,
32926
32927         // private
32928         register : function(dlg){
32929             list[dlg.id] = dlg;
32930             accessList.push(dlg);
32931         },
32932
32933         // private
32934         unregister : function(dlg){
32935             delete list[dlg.id];
32936             var i=0;
32937             var len=0;
32938             if(!accessList.indexOf){
32939                 for(  i = 0, len = accessList.length; i < len; i++){
32940                     if(accessList[i] == dlg){
32941                         accessList.splice(i, 1);
32942                         return;
32943                     }
32944                 }
32945             }else{
32946                  i = accessList.indexOf(dlg);
32947                 if(i != -1){
32948                     accessList.splice(i, 1);
32949                 }
32950             }
32951         },
32952
32953         /**
32954          * Gets a registered dialog by id
32955          * @param {String/Object} id The id of the dialog or a dialog
32956          * @return {Roo.BasicDialog} this
32957          */
32958         get : function(id){
32959             return typeof id == "object" ? id : list[id];
32960         },
32961
32962         /**
32963          * Brings the specified dialog to the front
32964          * @param {String/Object} dlg The id of the dialog or a dialog
32965          * @return {Roo.BasicDialog} this
32966          */
32967         bringToFront : function(dlg){
32968             dlg = this.get(dlg);
32969             if(dlg != front){
32970                 front = dlg;
32971                 dlg._lastAccess = new Date().getTime();
32972                 orderDialogs();
32973             }
32974             return dlg;
32975         },
32976
32977         /**
32978          * Sends the specified dialog to the back
32979          * @param {String/Object} dlg The id of the dialog or a dialog
32980          * @return {Roo.BasicDialog} this
32981          */
32982         sendToBack : function(dlg){
32983             dlg = this.get(dlg);
32984             dlg._lastAccess = -(new Date().getTime());
32985             orderDialogs();
32986             return dlg;
32987         },
32988
32989         /**
32990          * Hides all dialogs
32991          */
32992         hideAll : function(){
32993             for(var id in list){
32994                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32995                     list[id].hide();
32996                 }
32997             }
32998         }
32999     };
33000 }();
33001
33002 /**
33003  * @class Roo.LayoutDialog
33004  * @extends Roo.BasicDialog
33005  * Dialog which provides adjustments for working with a layout in a Dialog.
33006  * Add your necessary layout config options to the dialog's config.<br>
33007  * Example usage (including a nested layout):
33008  * <pre><code>
33009 if(!dialog){
33010     dialog = new Roo.LayoutDialog("download-dlg", {
33011         modal: true,
33012         width:600,
33013         height:450,
33014         shadow:true,
33015         minWidth:500,
33016         minHeight:350,
33017         autoTabs:true,
33018         proxyDrag:true,
33019         // layout config merges with the dialog config
33020         center:{
33021             tabPosition: "top",
33022             alwaysShowTabs: true
33023         }
33024     });
33025     dialog.addKeyListener(27, dialog.hide, dialog);
33026     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33027     dialog.addButton("Build It!", this.getDownload, this);
33028
33029     // we can even add nested layouts
33030     var innerLayout = new Roo.BorderLayout("dl-inner", {
33031         east: {
33032             initialSize: 200,
33033             autoScroll:true,
33034             split:true
33035         },
33036         center: {
33037             autoScroll:true
33038         }
33039     });
33040     innerLayout.beginUpdate();
33041     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33042     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33043     innerLayout.endUpdate(true);
33044
33045     var layout = dialog.getLayout();
33046     layout.beginUpdate();
33047     layout.add("center", new Roo.ContentPanel("standard-panel",
33048                         {title: "Download the Source", fitToFrame:true}));
33049     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33050                {title: "Build your own roo.js"}));
33051     layout.getRegion("center").showPanel(sp);
33052     layout.endUpdate();
33053 }
33054 </code></pre>
33055     * @constructor
33056     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33057     * @param {Object} config configuration options
33058   */
33059 Roo.LayoutDialog = function(el, cfg){
33060     
33061     var config=  cfg;
33062     if (typeof(cfg) == 'undefined') {
33063         config = Roo.apply({}, el);
33064         // not sure why we use documentElement here.. - it should always be body.
33065         // IE7 borks horribly if we use documentElement.
33066         // webkit also does not like documentElement - it creates a body element...
33067         el = Roo.get( document.body || document.documentElement ).createChild();
33068         //config.autoCreate = true;
33069     }
33070     
33071     
33072     config.autoTabs = false;
33073     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33074     this.body.setStyle({overflow:"hidden", position:"relative"});
33075     this.layout = new Roo.BorderLayout(this.body.dom, config);
33076     this.layout.monitorWindowResize = false;
33077     this.el.addClass("x-dlg-auto-layout");
33078     // fix case when center region overwrites center function
33079     this.center = Roo.BasicDialog.prototype.center;
33080     this.on("show", this.layout.layout, this.layout, true);
33081     if (config.items) {
33082         var xitems = config.items;
33083         delete config.items;
33084         Roo.each(xitems, this.addxtype, this);
33085     }
33086     
33087     
33088 };
33089 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33090     /**
33091      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33092      * @deprecated
33093      */
33094     endUpdate : function(){
33095         this.layout.endUpdate();
33096     },
33097
33098     /**
33099      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33100      *  @deprecated
33101      */
33102     beginUpdate : function(){
33103         this.layout.beginUpdate();
33104     },
33105
33106     /**
33107      * Get the BorderLayout for this dialog
33108      * @return {Roo.BorderLayout}
33109      */
33110     getLayout : function(){
33111         return this.layout;
33112     },
33113
33114     showEl : function(){
33115         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33116         if(Roo.isIE7){
33117             this.layout.layout();
33118         }
33119     },
33120
33121     // private
33122     // Use the syncHeightBeforeShow config option to control this automatically
33123     syncBodyHeight : function(){
33124         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33125         if(this.layout){this.layout.layout();}
33126     },
33127     
33128       /**
33129      * Add an xtype element (actually adds to the layout.)
33130      * @return {Object} xdata xtype object data.
33131      */
33132     
33133     addxtype : function(c) {
33134         return this.layout.addxtype(c);
33135     }
33136 });/*
33137  * Based on:
33138  * Ext JS Library 1.1.1
33139  * Copyright(c) 2006-2007, Ext JS, LLC.
33140  *
33141  * Originally Released Under LGPL - original licence link has changed is not relivant.
33142  *
33143  * Fork - LGPL
33144  * <script type="text/javascript">
33145  */
33146  
33147 /**
33148  * @class Roo.MessageBox
33149  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33150  * Example usage:
33151  *<pre><code>
33152 // Basic alert:
33153 Roo.Msg.alert('Status', 'Changes saved successfully.');
33154
33155 // Prompt for user data:
33156 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33157     if (btn == 'ok'){
33158         // process text value...
33159     }
33160 });
33161
33162 // Show a dialog using config options:
33163 Roo.Msg.show({
33164    title:'Save Changes?',
33165    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33166    buttons: Roo.Msg.YESNOCANCEL,
33167    fn: processResult,
33168    animEl: 'elId'
33169 });
33170 </code></pre>
33171  * @singleton
33172  */
33173 Roo.MessageBox = function(){
33174     var dlg, opt, mask, waitTimer;
33175     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33176     var buttons, activeTextEl, bwidth;
33177
33178     // private
33179     var handleButton = function(button){
33180         dlg.hide();
33181         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33182     };
33183
33184     // private
33185     var handleHide = function(){
33186         if(opt && opt.cls){
33187             dlg.el.removeClass(opt.cls);
33188         }
33189         if(waitTimer){
33190             Roo.TaskMgr.stop(waitTimer);
33191             waitTimer = null;
33192         }
33193     };
33194
33195     // private
33196     var updateButtons = function(b){
33197         var width = 0;
33198         if(!b){
33199             buttons["ok"].hide();
33200             buttons["cancel"].hide();
33201             buttons["yes"].hide();
33202             buttons["no"].hide();
33203             dlg.footer.dom.style.display = 'none';
33204             return width;
33205         }
33206         dlg.footer.dom.style.display = '';
33207         for(var k in buttons){
33208             if(typeof buttons[k] != "function"){
33209                 if(b[k]){
33210                     buttons[k].show();
33211                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33212                     width += buttons[k].el.getWidth()+15;
33213                 }else{
33214                     buttons[k].hide();
33215                 }
33216             }
33217         }
33218         return width;
33219     };
33220
33221     // private
33222     var handleEsc = function(d, k, e){
33223         if(opt && opt.closable !== false){
33224             dlg.hide();
33225         }
33226         if(e){
33227             e.stopEvent();
33228         }
33229     };
33230
33231     return {
33232         /**
33233          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33234          * @return {Roo.BasicDialog} The BasicDialog element
33235          */
33236         getDialog : function(){
33237            if(!dlg){
33238                 dlg = new Roo.BasicDialog("x-msg-box", {
33239                     autoCreate : true,
33240                     shadow: true,
33241                     draggable: true,
33242                     resizable:false,
33243                     constraintoviewport:false,
33244                     fixedcenter:true,
33245                     collapsible : false,
33246                     shim:true,
33247                     modal: true,
33248                     width:400, height:100,
33249                     buttonAlign:"center",
33250                     closeClick : function(){
33251                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33252                             handleButton("no");
33253                         }else{
33254                             handleButton("cancel");
33255                         }
33256                     }
33257                 });
33258                 dlg.on("hide", handleHide);
33259                 mask = dlg.mask;
33260                 dlg.addKeyListener(27, handleEsc);
33261                 buttons = {};
33262                 var bt = this.buttonText;
33263                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33264                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33265                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33266                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33267                 bodyEl = dlg.body.createChild({
33268
33269                     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>'
33270                 });
33271                 msgEl = bodyEl.dom.firstChild;
33272                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33273                 textboxEl.enableDisplayMode();
33274                 textboxEl.addKeyListener([10,13], function(){
33275                     if(dlg.isVisible() && opt && opt.buttons){
33276                         if(opt.buttons.ok){
33277                             handleButton("ok");
33278                         }else if(opt.buttons.yes){
33279                             handleButton("yes");
33280                         }
33281                     }
33282                 });
33283                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33284                 textareaEl.enableDisplayMode();
33285                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33286                 progressEl.enableDisplayMode();
33287                 var pf = progressEl.dom.firstChild;
33288                 if (pf) {
33289                     pp = Roo.get(pf.firstChild);
33290                     pp.setHeight(pf.offsetHeight);
33291                 }
33292                 
33293             }
33294             return dlg;
33295         },
33296
33297         /**
33298          * Updates the message box body text
33299          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33300          * the XHTML-compliant non-breaking space character '&amp;#160;')
33301          * @return {Roo.MessageBox} This message box
33302          */
33303         updateText : function(text){
33304             if(!dlg.isVisible() && !opt.width){
33305                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33306             }
33307             msgEl.innerHTML = text || '&#160;';
33308       
33309             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33310             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33311             var w = Math.max(
33312                     Math.min(opt.width || cw , this.maxWidth), 
33313                     Math.max(opt.minWidth || this.minWidth, bwidth)
33314             );
33315             if(opt.prompt){
33316                 activeTextEl.setWidth(w);
33317             }
33318             if(dlg.isVisible()){
33319                 dlg.fixedcenter = false;
33320             }
33321             // to big, make it scroll. = But as usual stupid IE does not support
33322             // !important..
33323             
33324             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33325                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33326                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33327             } else {
33328                 bodyEl.dom.style.height = '';
33329                 bodyEl.dom.style.overflowY = '';
33330             }
33331             if (cw > w) {
33332                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33333             } else {
33334                 bodyEl.dom.style.overflowX = '';
33335             }
33336             
33337             dlg.setContentSize(w, bodyEl.getHeight());
33338             if(dlg.isVisible()){
33339                 dlg.fixedcenter = true;
33340             }
33341             return this;
33342         },
33343
33344         /**
33345          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33346          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33347          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33348          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33349          * @return {Roo.MessageBox} This message box
33350          */
33351         updateProgress : function(value, text){
33352             if(text){
33353                 this.updateText(text);
33354             }
33355             if (pp) { // weird bug on my firefox - for some reason this is not defined
33356                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33357             }
33358             return this;
33359         },        
33360
33361         /**
33362          * Returns true if the message box is currently displayed
33363          * @return {Boolean} True if the message box is visible, else false
33364          */
33365         isVisible : function(){
33366             return dlg && dlg.isVisible();  
33367         },
33368
33369         /**
33370          * Hides the message box if it is displayed
33371          */
33372         hide : function(){
33373             if(this.isVisible()){
33374                 dlg.hide();
33375             }  
33376         },
33377
33378         /**
33379          * Displays a new message box, or reinitializes an existing message box, based on the config options
33380          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33381          * The following config object properties are supported:
33382          * <pre>
33383 Property    Type             Description
33384 ----------  ---------------  ------------------------------------------------------------------------------------
33385 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33386                                    closes (defaults to undefined)
33387 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33388                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33389 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33390                                    progress and wait dialogs will ignore this property and always hide the
33391                                    close button as they can only be closed programmatically.
33392 cls               String           A custom CSS class to apply to the message box element
33393 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33394                                    displayed (defaults to 75)
33395 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33396                                    function will be btn (the name of the button that was clicked, if applicable,
33397                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33398                                    Progress and wait dialogs will ignore this option since they do not respond to
33399                                    user actions and can only be closed programmatically, so any required function
33400                                    should be called by the same code after it closes the dialog.
33401 icon              String           A CSS class that provides a background image to be used as an icon for
33402                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33403 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33404 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33405 modal             Boolean          False to allow user interaction with the page while the message box is
33406                                    displayed (defaults to true)
33407 msg               String           A string that will replace the existing message box body text (defaults
33408                                    to the XHTML-compliant non-breaking space character '&#160;')
33409 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33410 progress          Boolean          True to display a progress bar (defaults to false)
33411 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33412 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33413 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33414 title             String           The title text
33415 value             String           The string value to set into the active textbox element if displayed
33416 wait              Boolean          True to display a progress bar (defaults to false)
33417 width             Number           The width of the dialog in pixels
33418 </pre>
33419          *
33420          * Example usage:
33421          * <pre><code>
33422 Roo.Msg.show({
33423    title: 'Address',
33424    msg: 'Please enter your address:',
33425    width: 300,
33426    buttons: Roo.MessageBox.OKCANCEL,
33427    multiline: true,
33428    fn: saveAddress,
33429    animEl: 'addAddressBtn'
33430 });
33431 </code></pre>
33432          * @param {Object} config Configuration options
33433          * @return {Roo.MessageBox} This message box
33434          */
33435         show : function(options)
33436         {
33437             
33438             // this causes nightmares if you show one dialog after another
33439             // especially on callbacks..
33440              
33441             if(this.isVisible()){
33442                 
33443                 this.hide();
33444                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33445                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33446                 Roo.log("New Dialog Message:" +  options.msg )
33447                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33448                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33449                 
33450             }
33451             var d = this.getDialog();
33452             opt = options;
33453             d.setTitle(opt.title || "&#160;");
33454             d.close.setDisplayed(opt.closable !== false);
33455             activeTextEl = textboxEl;
33456             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33457             if(opt.prompt){
33458                 if(opt.multiline){
33459                     textboxEl.hide();
33460                     textareaEl.show();
33461                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33462                         opt.multiline : this.defaultTextHeight);
33463                     activeTextEl = textareaEl;
33464                 }else{
33465                     textboxEl.show();
33466                     textareaEl.hide();
33467                 }
33468             }else{
33469                 textboxEl.hide();
33470                 textareaEl.hide();
33471             }
33472             progressEl.setDisplayed(opt.progress === true);
33473             this.updateProgress(0);
33474             activeTextEl.dom.value = opt.value || "";
33475             if(opt.prompt){
33476                 dlg.setDefaultButton(activeTextEl);
33477             }else{
33478                 var bs = opt.buttons;
33479                 var db = null;
33480                 if(bs && bs.ok){
33481                     db = buttons["ok"];
33482                 }else if(bs && bs.yes){
33483                     db = buttons["yes"];
33484                 }
33485                 dlg.setDefaultButton(db);
33486             }
33487             bwidth = updateButtons(opt.buttons);
33488             this.updateText(opt.msg);
33489             if(opt.cls){
33490                 d.el.addClass(opt.cls);
33491             }
33492             d.proxyDrag = opt.proxyDrag === true;
33493             d.modal = opt.modal !== false;
33494             d.mask = opt.modal !== false ? mask : false;
33495             if(!d.isVisible()){
33496                 // force it to the end of the z-index stack so it gets a cursor in FF
33497                 document.body.appendChild(dlg.el.dom);
33498                 d.animateTarget = null;
33499                 d.show(options.animEl);
33500             }
33501             return this;
33502         },
33503
33504         /**
33505          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33506          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33507          * and closing the message box when the process is complete.
33508          * @param {String} title The title bar text
33509          * @param {String} msg The message box body text
33510          * @return {Roo.MessageBox} This message box
33511          */
33512         progress : function(title, msg){
33513             this.show({
33514                 title : title,
33515                 msg : msg,
33516                 buttons: false,
33517                 progress:true,
33518                 closable:false,
33519                 minWidth: this.minProgressWidth,
33520                 modal : true
33521             });
33522             return this;
33523         },
33524
33525         /**
33526          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33527          * If a callback function is passed it will be called after the user clicks the button, and the
33528          * id of the button that was clicked will be passed as the only parameter to the callback
33529          * (could also be the top-right close button).
33530          * @param {String} title The title bar text
33531          * @param {String} msg The message box body text
33532          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33533          * @param {Object} scope (optional) The scope of the callback function
33534          * @return {Roo.MessageBox} This message box
33535          */
33536         alert : function(title, msg, fn, scope){
33537             this.show({
33538                 title : title,
33539                 msg : msg,
33540                 buttons: this.OK,
33541                 fn: fn,
33542                 scope : scope,
33543                 modal : true
33544             });
33545             return this;
33546         },
33547
33548         /**
33549          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33550          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33551          * You are responsible for closing the message box when the process is complete.
33552          * @param {String} msg The message box body text
33553          * @param {String} title (optional) The title bar text
33554          * @return {Roo.MessageBox} This message box
33555          */
33556         wait : function(msg, title){
33557             this.show({
33558                 title : title,
33559                 msg : msg,
33560                 buttons: false,
33561                 closable:false,
33562                 progress:true,
33563                 modal:true,
33564                 width:300,
33565                 wait:true
33566             });
33567             waitTimer = Roo.TaskMgr.start({
33568                 run: function(i){
33569                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33570                 },
33571                 interval: 1000
33572             });
33573             return this;
33574         },
33575
33576         /**
33577          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33578          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33579          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33580          * @param {String} title The title bar text
33581          * @param {String} msg The message box body text
33582          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33583          * @param {Object} scope (optional) The scope of the callback function
33584          * @return {Roo.MessageBox} This message box
33585          */
33586         confirm : function(title, msg, fn, scope){
33587             this.show({
33588                 title : title,
33589                 msg : msg,
33590                 buttons: this.YESNO,
33591                 fn: fn,
33592                 scope : scope,
33593                 modal : true
33594             });
33595             return this;
33596         },
33597
33598         /**
33599          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33600          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33601          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33602          * (could also be the top-right close button) and the text that was entered will be passed as the two
33603          * parameters to the callback.
33604          * @param {String} title The title bar text
33605          * @param {String} msg The message box body text
33606          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33607          * @param {Object} scope (optional) The scope of the callback function
33608          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33609          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33610          * @return {Roo.MessageBox} This message box
33611          */
33612         prompt : function(title, msg, fn, scope, multiline){
33613             this.show({
33614                 title : title,
33615                 msg : msg,
33616                 buttons: this.OKCANCEL,
33617                 fn: fn,
33618                 minWidth:250,
33619                 scope : scope,
33620                 prompt:true,
33621                 multiline: multiline,
33622                 modal : true
33623             });
33624             return this;
33625         },
33626
33627         /**
33628          * Button config that displays a single OK button
33629          * @type Object
33630          */
33631         OK : {ok:true},
33632         /**
33633          * Button config that displays Yes and No buttons
33634          * @type Object
33635          */
33636         YESNO : {yes:true, no:true},
33637         /**
33638          * Button config that displays OK and Cancel buttons
33639          * @type Object
33640          */
33641         OKCANCEL : {ok:true, cancel:true},
33642         /**
33643          * Button config that displays Yes, No and Cancel buttons
33644          * @type Object
33645          */
33646         YESNOCANCEL : {yes:true, no:true, cancel:true},
33647
33648         /**
33649          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33650          * @type Number
33651          */
33652         defaultTextHeight : 75,
33653         /**
33654          * The maximum width in pixels of the message box (defaults to 600)
33655          * @type Number
33656          */
33657         maxWidth : 600,
33658         /**
33659          * The minimum width in pixels of the message box (defaults to 100)
33660          * @type Number
33661          */
33662         minWidth : 100,
33663         /**
33664          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33665          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33666          * @type Number
33667          */
33668         minProgressWidth : 250,
33669         /**
33670          * An object containing the default button text strings that can be overriden for localized language support.
33671          * Supported properties are: ok, cancel, yes and no.
33672          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33673          * @type Object
33674          */
33675         buttonText : {
33676             ok : "OK",
33677             cancel : "Cancel",
33678             yes : "Yes",
33679             no : "No"
33680         }
33681     };
33682 }();
33683
33684 /**
33685  * Shorthand for {@link Roo.MessageBox}
33686  */
33687 Roo.Msg = Roo.MessageBox;/*
33688  * Based on:
33689  * Ext JS Library 1.1.1
33690  * Copyright(c) 2006-2007, Ext JS, LLC.
33691  *
33692  * Originally Released Under LGPL - original licence link has changed is not relivant.
33693  *
33694  * Fork - LGPL
33695  * <script type="text/javascript">
33696  */
33697 /**
33698  * @class Roo.QuickTips
33699  * Provides attractive and customizable tooltips for any element.
33700  * @singleton
33701  */
33702 Roo.QuickTips = function(){
33703     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33704     var ce, bd, xy, dd;
33705     var visible = false, disabled = true, inited = false;
33706     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33707     
33708     var onOver = function(e){
33709         if(disabled){
33710             return;
33711         }
33712         var t = e.getTarget();
33713         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33714             return;
33715         }
33716         if(ce && t == ce.el){
33717             clearTimeout(hideProc);
33718             return;
33719         }
33720         if(t && tagEls[t.id]){
33721             tagEls[t.id].el = t;
33722             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33723             return;
33724         }
33725         var ttp, et = Roo.fly(t);
33726         var ns = cfg.namespace;
33727         if(tm.interceptTitles && t.title){
33728             ttp = t.title;
33729             t.qtip = ttp;
33730             t.removeAttribute("title");
33731             e.preventDefault();
33732         }else{
33733             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33734         }
33735         if(ttp){
33736             showProc = show.defer(tm.showDelay, tm, [{
33737                 el: t, 
33738                 text: ttp.replace(/\\n/g,'<br/>'),
33739                 width: et.getAttributeNS(ns, cfg.width),
33740                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33741                 title: et.getAttributeNS(ns, cfg.title),
33742                     cls: et.getAttributeNS(ns, cfg.cls)
33743             }]);
33744         }
33745     };
33746     
33747     var onOut = function(e){
33748         clearTimeout(showProc);
33749         var t = e.getTarget();
33750         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33751             hideProc = setTimeout(hide, tm.hideDelay);
33752         }
33753     };
33754     
33755     var onMove = function(e){
33756         if(disabled){
33757             return;
33758         }
33759         xy = e.getXY();
33760         xy[1] += 18;
33761         if(tm.trackMouse && ce){
33762             el.setXY(xy);
33763         }
33764     };
33765     
33766     var onDown = function(e){
33767         clearTimeout(showProc);
33768         clearTimeout(hideProc);
33769         if(!e.within(el)){
33770             if(tm.hideOnClick){
33771                 hide();
33772                 tm.disable();
33773                 tm.enable.defer(100, tm);
33774             }
33775         }
33776     };
33777     
33778     var getPad = function(){
33779         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33780     };
33781
33782     var show = function(o){
33783         if(disabled){
33784             return;
33785         }
33786         clearTimeout(dismissProc);
33787         ce = o;
33788         if(removeCls){ // in case manually hidden
33789             el.removeClass(removeCls);
33790             removeCls = null;
33791         }
33792         if(ce.cls){
33793             el.addClass(ce.cls);
33794             removeCls = ce.cls;
33795         }
33796         if(ce.title){
33797             tipTitle.update(ce.title);
33798             tipTitle.show();
33799         }else{
33800             tipTitle.update('');
33801             tipTitle.hide();
33802         }
33803         el.dom.style.width  = tm.maxWidth+'px';
33804         //tipBody.dom.style.width = '';
33805         tipBodyText.update(o.text);
33806         var p = getPad(), w = ce.width;
33807         if(!w){
33808             var td = tipBodyText.dom;
33809             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33810             if(aw > tm.maxWidth){
33811                 w = tm.maxWidth;
33812             }else if(aw < tm.minWidth){
33813                 w = tm.minWidth;
33814             }else{
33815                 w = aw;
33816             }
33817         }
33818         //tipBody.setWidth(w);
33819         el.setWidth(parseInt(w, 10) + p);
33820         if(ce.autoHide === false){
33821             close.setDisplayed(true);
33822             if(dd){
33823                 dd.unlock();
33824             }
33825         }else{
33826             close.setDisplayed(false);
33827             if(dd){
33828                 dd.lock();
33829             }
33830         }
33831         if(xy){
33832             el.avoidY = xy[1]-18;
33833             el.setXY(xy);
33834         }
33835         if(tm.animate){
33836             el.setOpacity(.1);
33837             el.setStyle("visibility", "visible");
33838             el.fadeIn({callback: afterShow});
33839         }else{
33840             afterShow();
33841         }
33842     };
33843     
33844     var afterShow = function(){
33845         if(ce){
33846             el.show();
33847             esc.enable();
33848             if(tm.autoDismiss && ce.autoHide !== false){
33849                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33850             }
33851         }
33852     };
33853     
33854     var hide = function(noanim){
33855         clearTimeout(dismissProc);
33856         clearTimeout(hideProc);
33857         ce = null;
33858         if(el.isVisible()){
33859             esc.disable();
33860             if(noanim !== true && tm.animate){
33861                 el.fadeOut({callback: afterHide});
33862             }else{
33863                 afterHide();
33864             } 
33865         }
33866     };
33867     
33868     var afterHide = function(){
33869         el.hide();
33870         if(removeCls){
33871             el.removeClass(removeCls);
33872             removeCls = null;
33873         }
33874     };
33875     
33876     return {
33877         /**
33878         * @cfg {Number} minWidth
33879         * The minimum width of the quick tip (defaults to 40)
33880         */
33881        minWidth : 40,
33882         /**
33883         * @cfg {Number} maxWidth
33884         * The maximum width of the quick tip (defaults to 300)
33885         */
33886        maxWidth : 300,
33887         /**
33888         * @cfg {Boolean} interceptTitles
33889         * True to automatically use the element's DOM title value if available (defaults to false)
33890         */
33891        interceptTitles : false,
33892         /**
33893         * @cfg {Boolean} trackMouse
33894         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33895         */
33896        trackMouse : false,
33897         /**
33898         * @cfg {Boolean} hideOnClick
33899         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33900         */
33901        hideOnClick : true,
33902         /**
33903         * @cfg {Number} showDelay
33904         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33905         */
33906        showDelay : 500,
33907         /**
33908         * @cfg {Number} hideDelay
33909         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33910         */
33911        hideDelay : 200,
33912         /**
33913         * @cfg {Boolean} autoHide
33914         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33915         * Used in conjunction with hideDelay.
33916         */
33917        autoHide : true,
33918         /**
33919         * @cfg {Boolean}
33920         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33921         * (defaults to true).  Used in conjunction with autoDismissDelay.
33922         */
33923        autoDismiss : true,
33924         /**
33925         * @cfg {Number}
33926         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33927         */
33928        autoDismissDelay : 5000,
33929        /**
33930         * @cfg {Boolean} animate
33931         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33932         */
33933        animate : false,
33934
33935        /**
33936         * @cfg {String} title
33937         * Title text to display (defaults to '').  This can be any valid HTML markup.
33938         */
33939         title: '',
33940        /**
33941         * @cfg {String} text
33942         * Body text to display (defaults to '').  This can be any valid HTML markup.
33943         */
33944         text : '',
33945        /**
33946         * @cfg {String} cls
33947         * A CSS class to apply to the base quick tip element (defaults to '').
33948         */
33949         cls : '',
33950        /**
33951         * @cfg {Number} width
33952         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33953         * minWidth or maxWidth.
33954         */
33955         width : null,
33956
33957     /**
33958      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33959      * or display QuickTips in a page.
33960      */
33961        init : function(){
33962           tm = Roo.QuickTips;
33963           cfg = tm.tagConfig;
33964           if(!inited){
33965               if(!Roo.isReady){ // allow calling of init() before onReady
33966                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33967                   return;
33968               }
33969               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33970               el.fxDefaults = {stopFx: true};
33971               // maximum custom styling
33972               //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>');
33973               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>');              
33974               tipTitle = el.child('h3');
33975               tipTitle.enableDisplayMode("block");
33976               tipBody = el.child('div.x-tip-bd');
33977               tipBodyText = el.child('div.x-tip-bd-inner');
33978               //bdLeft = el.child('div.x-tip-bd-left');
33979               //bdRight = el.child('div.x-tip-bd-right');
33980               close = el.child('div.x-tip-close');
33981               close.enableDisplayMode("block");
33982               close.on("click", hide);
33983               var d = Roo.get(document);
33984               d.on("mousedown", onDown);
33985               d.on("mouseover", onOver);
33986               d.on("mouseout", onOut);
33987               d.on("mousemove", onMove);
33988               esc = d.addKeyListener(27, hide);
33989               esc.disable();
33990               if(Roo.dd.DD){
33991                   dd = el.initDD("default", null, {
33992                       onDrag : function(){
33993                           el.sync();  
33994                       }
33995                   });
33996                   dd.setHandleElId(tipTitle.id);
33997                   dd.lock();
33998               }
33999               inited = true;
34000           }
34001           this.enable(); 
34002        },
34003
34004     /**
34005      * Configures a new quick tip instance and assigns it to a target element.  The following config options
34006      * are supported:
34007      * <pre>
34008 Property    Type                   Description
34009 ----------  ---------------------  ------------------------------------------------------------------------
34010 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34011      * </ul>
34012      * @param {Object} config The config object
34013      */
34014        register : function(config){
34015            var cs = config instanceof Array ? config : arguments;
34016            for(var i = 0, len = cs.length; i < len; i++) {
34017                var c = cs[i];
34018                var target = c.target;
34019                if(target){
34020                    if(target instanceof Array){
34021                        for(var j = 0, jlen = target.length; j < jlen; j++){
34022                            tagEls[target[j]] = c;
34023                        }
34024                    }else{
34025                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34026                    }
34027                }
34028            }
34029        },
34030
34031     /**
34032      * Removes this quick tip from its element and destroys it.
34033      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34034      */
34035        unregister : function(el){
34036            delete tagEls[Roo.id(el)];
34037        },
34038
34039     /**
34040      * Enable this quick tip.
34041      */
34042        enable : function(){
34043            if(inited && disabled){
34044                locks.pop();
34045                if(locks.length < 1){
34046                    disabled = false;
34047                }
34048            }
34049        },
34050
34051     /**
34052      * Disable this quick tip.
34053      */
34054        disable : function(){
34055           disabled = true;
34056           clearTimeout(showProc);
34057           clearTimeout(hideProc);
34058           clearTimeout(dismissProc);
34059           if(ce){
34060               hide(true);
34061           }
34062           locks.push(1);
34063        },
34064
34065     /**
34066      * Returns true if the quick tip is enabled, else false.
34067      */
34068        isEnabled : function(){
34069             return !disabled;
34070        },
34071
34072         // private
34073        tagConfig : {
34074            namespace : "roo", // was ext?? this may break..
34075            alt_namespace : "ext",
34076            attribute : "qtip",
34077            width : "width",
34078            target : "target",
34079            title : "qtitle",
34080            hide : "hide",
34081            cls : "qclass"
34082        }
34083    };
34084 }();
34085
34086 // backwards compat
34087 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34088  * Based on:
34089  * Ext JS Library 1.1.1
34090  * Copyright(c) 2006-2007, Ext JS, LLC.
34091  *
34092  * Originally Released Under LGPL - original licence link has changed is not relivant.
34093  *
34094  * Fork - LGPL
34095  * <script type="text/javascript">
34096  */
34097  
34098
34099 /**
34100  * @class Roo.tree.TreePanel
34101  * @extends Roo.data.Tree
34102
34103  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34104  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34105  * @cfg {Boolean} enableDD true to enable drag and drop
34106  * @cfg {Boolean} enableDrag true to enable just drag
34107  * @cfg {Boolean} enableDrop true to enable just drop
34108  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34109  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34110  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34111  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34112  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34113  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34114  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34115  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34116  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34117  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34118  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34119  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34120  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34121  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34122  * @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>
34123  * @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>
34124  * 
34125  * @constructor
34126  * @param {String/HTMLElement/Element} el The container element
34127  * @param {Object} config
34128  */
34129 Roo.tree.TreePanel = function(el, config){
34130     var root = false;
34131     var loader = false;
34132     if (config.root) {
34133         root = config.root;
34134         delete config.root;
34135     }
34136     if (config.loader) {
34137         loader = config.loader;
34138         delete config.loader;
34139     }
34140     
34141     Roo.apply(this, config);
34142     Roo.tree.TreePanel.superclass.constructor.call(this);
34143     this.el = Roo.get(el);
34144     this.el.addClass('x-tree');
34145     //console.log(root);
34146     if (root) {
34147         this.setRootNode( Roo.factory(root, Roo.tree));
34148     }
34149     if (loader) {
34150         this.loader = Roo.factory(loader, Roo.tree);
34151     }
34152    /**
34153     * Read-only. The id of the container element becomes this TreePanel's id.
34154     */
34155     this.id = this.el.id;
34156     this.addEvents({
34157         /**
34158         * @event beforeload
34159         * Fires before a node is loaded, return false to cancel
34160         * @param {Node} node The node being loaded
34161         */
34162         "beforeload" : true,
34163         /**
34164         * @event load
34165         * Fires when a node is loaded
34166         * @param {Node} node The node that was loaded
34167         */
34168         "load" : true,
34169         /**
34170         * @event textchange
34171         * Fires when the text for a node is changed
34172         * @param {Node} node The node
34173         * @param {String} text The new text
34174         * @param {String} oldText The old text
34175         */
34176         "textchange" : true,
34177         /**
34178         * @event beforeexpand
34179         * Fires before a node is expanded, return false to cancel.
34180         * @param {Node} node The node
34181         * @param {Boolean} deep
34182         * @param {Boolean} anim
34183         */
34184         "beforeexpand" : true,
34185         /**
34186         * @event beforecollapse
34187         * Fires before a node is collapsed, return false to cancel.
34188         * @param {Node} node The node
34189         * @param {Boolean} deep
34190         * @param {Boolean} anim
34191         */
34192         "beforecollapse" : true,
34193         /**
34194         * @event expand
34195         * Fires when a node is expanded
34196         * @param {Node} node The node
34197         */
34198         "expand" : true,
34199         /**
34200         * @event disabledchange
34201         * Fires when the disabled status of a node changes
34202         * @param {Node} node The node
34203         * @param {Boolean} disabled
34204         */
34205         "disabledchange" : true,
34206         /**
34207         * @event collapse
34208         * Fires when a node is collapsed
34209         * @param {Node} node The node
34210         */
34211         "collapse" : true,
34212         /**
34213         * @event beforeclick
34214         * Fires before click processing on a node. Return false to cancel the default action.
34215         * @param {Node} node The node
34216         * @param {Roo.EventObject} e The event object
34217         */
34218         "beforeclick":true,
34219         /**
34220         * @event checkchange
34221         * Fires when a node with a checkbox's checked property changes
34222         * @param {Node} this This node
34223         * @param {Boolean} checked
34224         */
34225         "checkchange":true,
34226         /**
34227         * @event click
34228         * Fires when a node is clicked
34229         * @param {Node} node The node
34230         * @param {Roo.EventObject} e The event object
34231         */
34232         "click":true,
34233         /**
34234         * @event dblclick
34235         * Fires when a node is double clicked
34236         * @param {Node} node The node
34237         * @param {Roo.EventObject} e The event object
34238         */
34239         "dblclick":true,
34240         /**
34241         * @event contextmenu
34242         * Fires when a node is right clicked
34243         * @param {Node} node The node
34244         * @param {Roo.EventObject} e The event object
34245         */
34246         "contextmenu":true,
34247         /**
34248         * @event beforechildrenrendered
34249         * Fires right before the child nodes for a node are rendered
34250         * @param {Node} node The node
34251         */
34252         "beforechildrenrendered":true,
34253         /**
34254         * @event startdrag
34255         * Fires when a node starts being dragged
34256         * @param {Roo.tree.TreePanel} this
34257         * @param {Roo.tree.TreeNode} node
34258         * @param {event} e The raw browser event
34259         */ 
34260        "startdrag" : true,
34261        /**
34262         * @event enddrag
34263         * Fires when a drag operation is complete
34264         * @param {Roo.tree.TreePanel} this
34265         * @param {Roo.tree.TreeNode} node
34266         * @param {event} e The raw browser event
34267         */
34268        "enddrag" : true,
34269        /**
34270         * @event dragdrop
34271         * Fires when a dragged node is dropped on a valid DD target
34272         * @param {Roo.tree.TreePanel} this
34273         * @param {Roo.tree.TreeNode} node
34274         * @param {DD} dd The dd it was dropped on
34275         * @param {event} e The raw browser event
34276         */
34277        "dragdrop" : true,
34278        /**
34279         * @event beforenodedrop
34280         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34281         * passed to handlers has the following properties:<br />
34282         * <ul style="padding:5px;padding-left:16px;">
34283         * <li>tree - The TreePanel</li>
34284         * <li>target - The node being targeted for the drop</li>
34285         * <li>data - The drag data from the drag source</li>
34286         * <li>point - The point of the drop - append, above or below</li>
34287         * <li>source - The drag source</li>
34288         * <li>rawEvent - Raw mouse event</li>
34289         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34290         * to be inserted by setting them on this object.</li>
34291         * <li>cancel - Set this to true to cancel the drop.</li>
34292         * </ul>
34293         * @param {Object} dropEvent
34294         */
34295        "beforenodedrop" : true,
34296        /**
34297         * @event nodedrop
34298         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34299         * passed to handlers has the following properties:<br />
34300         * <ul style="padding:5px;padding-left:16px;">
34301         * <li>tree - The TreePanel</li>
34302         * <li>target - The node being targeted for the drop</li>
34303         * <li>data - The drag data from the drag source</li>
34304         * <li>point - The point of the drop - append, above or below</li>
34305         * <li>source - The drag source</li>
34306         * <li>rawEvent - Raw mouse event</li>
34307         * <li>dropNode - Dropped node(s).</li>
34308         * </ul>
34309         * @param {Object} dropEvent
34310         */
34311        "nodedrop" : true,
34312         /**
34313         * @event nodedragover
34314         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34315         * passed to handlers has the following properties:<br />
34316         * <ul style="padding:5px;padding-left:16px;">
34317         * <li>tree - The TreePanel</li>
34318         * <li>target - The node being targeted for the drop</li>
34319         * <li>data - The drag data from the drag source</li>
34320         * <li>point - The point of the drop - append, above or below</li>
34321         * <li>source - The drag source</li>
34322         * <li>rawEvent - Raw mouse event</li>
34323         * <li>dropNode - Drop node(s) provided by the source.</li>
34324         * <li>cancel - Set this to true to signal drop not allowed.</li>
34325         * </ul>
34326         * @param {Object} dragOverEvent
34327         */
34328        "nodedragover" : true,
34329        /**
34330         * @event appendnode
34331         * Fires when append node to the tree
34332         * @param {Roo.tree.TreePanel} this
34333         * @param {Roo.tree.TreeNode} node
34334         * @param {Number} index The index of the newly appended node
34335         */
34336        "appendnode" : true
34337         
34338     });
34339     if(this.singleExpand){
34340        this.on("beforeexpand", this.restrictExpand, this);
34341     }
34342     if (this.editor) {
34343         this.editor.tree = this;
34344         this.editor = Roo.factory(this.editor, Roo.tree);
34345     }
34346     
34347     if (this.selModel) {
34348         this.selModel = Roo.factory(this.selModel, Roo.tree);
34349     }
34350    
34351 };
34352 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34353     rootVisible : true,
34354     animate: Roo.enableFx,
34355     lines : true,
34356     enableDD : false,
34357     hlDrop : Roo.enableFx,
34358   
34359     renderer: false,
34360     
34361     rendererTip: false,
34362     // private
34363     restrictExpand : function(node){
34364         var p = node.parentNode;
34365         if(p){
34366             if(p.expandedChild && p.expandedChild.parentNode == p){
34367                 p.expandedChild.collapse();
34368             }
34369             p.expandedChild = node;
34370         }
34371     },
34372
34373     // private override
34374     setRootNode : function(node){
34375         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34376         if(!this.rootVisible){
34377             node.ui = new Roo.tree.RootTreeNodeUI(node);
34378         }
34379         return node;
34380     },
34381
34382     /**
34383      * Returns the container element for this TreePanel
34384      */
34385     getEl : function(){
34386         return this.el;
34387     },
34388
34389     /**
34390      * Returns the default TreeLoader for this TreePanel
34391      */
34392     getLoader : function(){
34393         return this.loader;
34394     },
34395
34396     /**
34397      * Expand all nodes
34398      */
34399     expandAll : function(){
34400         this.root.expand(true);
34401     },
34402
34403     /**
34404      * Collapse all nodes
34405      */
34406     collapseAll : function(){
34407         this.root.collapse(true);
34408     },
34409
34410     /**
34411      * Returns the selection model used by this TreePanel
34412      */
34413     getSelectionModel : function(){
34414         if(!this.selModel){
34415             this.selModel = new Roo.tree.DefaultSelectionModel();
34416         }
34417         return this.selModel;
34418     },
34419
34420     /**
34421      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34422      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34423      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34424      * @return {Array}
34425      */
34426     getChecked : function(a, startNode){
34427         startNode = startNode || this.root;
34428         var r = [];
34429         var f = function(){
34430             if(this.attributes.checked){
34431                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34432             }
34433         }
34434         startNode.cascade(f);
34435         return r;
34436     },
34437
34438     /**
34439      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34440      * @param {String} path
34441      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34442      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34443      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34444      */
34445     expandPath : function(path, attr, callback){
34446         attr = attr || "id";
34447         var keys = path.split(this.pathSeparator);
34448         var curNode = this.root;
34449         if(curNode.attributes[attr] != keys[1]){ // invalid root
34450             if(callback){
34451                 callback(false, null);
34452             }
34453             return;
34454         }
34455         var index = 1;
34456         var f = function(){
34457             if(++index == keys.length){
34458                 if(callback){
34459                     callback(true, curNode);
34460                 }
34461                 return;
34462             }
34463             var c = curNode.findChild(attr, keys[index]);
34464             if(!c){
34465                 if(callback){
34466                     callback(false, curNode);
34467                 }
34468                 return;
34469             }
34470             curNode = c;
34471             c.expand(false, false, f);
34472         };
34473         curNode.expand(false, false, f);
34474     },
34475
34476     /**
34477      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34478      * @param {String} path
34479      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34480      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34481      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34482      */
34483     selectPath : function(path, attr, callback){
34484         attr = attr || "id";
34485         var keys = path.split(this.pathSeparator);
34486         var v = keys.pop();
34487         if(keys.length > 0){
34488             var f = function(success, node){
34489                 if(success && node){
34490                     var n = node.findChild(attr, v);
34491                     if(n){
34492                         n.select();
34493                         if(callback){
34494                             callback(true, n);
34495                         }
34496                     }else if(callback){
34497                         callback(false, n);
34498                     }
34499                 }else{
34500                     if(callback){
34501                         callback(false, n);
34502                     }
34503                 }
34504             };
34505             this.expandPath(keys.join(this.pathSeparator), attr, f);
34506         }else{
34507             this.root.select();
34508             if(callback){
34509                 callback(true, this.root);
34510             }
34511         }
34512     },
34513
34514     getTreeEl : function(){
34515         return this.el;
34516     },
34517
34518     /**
34519      * Trigger rendering of this TreePanel
34520      */
34521     render : function(){
34522         if (this.innerCt) {
34523             return this; // stop it rendering more than once!!
34524         }
34525         
34526         this.innerCt = this.el.createChild({tag:"ul",
34527                cls:"x-tree-root-ct " +
34528                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34529
34530         if(this.containerScroll){
34531             Roo.dd.ScrollManager.register(this.el);
34532         }
34533         if((this.enableDD || this.enableDrop) && !this.dropZone){
34534            /**
34535             * The dropZone used by this tree if drop is enabled
34536             * @type Roo.tree.TreeDropZone
34537             */
34538              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34539                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34540            });
34541         }
34542         if((this.enableDD || this.enableDrag) && !this.dragZone){
34543            /**
34544             * The dragZone used by this tree if drag is enabled
34545             * @type Roo.tree.TreeDragZone
34546             */
34547             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34548                ddGroup: this.ddGroup || "TreeDD",
34549                scroll: this.ddScroll
34550            });
34551         }
34552         this.getSelectionModel().init(this);
34553         if (!this.root) {
34554             Roo.log("ROOT not set in tree");
34555             return this;
34556         }
34557         this.root.render();
34558         if(!this.rootVisible){
34559             this.root.renderChildren();
34560         }
34561         return this;
34562     }
34563 });/*
34564  * Based on:
34565  * Ext JS Library 1.1.1
34566  * Copyright(c) 2006-2007, Ext JS, LLC.
34567  *
34568  * Originally Released Under LGPL - original licence link has changed is not relivant.
34569  *
34570  * Fork - LGPL
34571  * <script type="text/javascript">
34572  */
34573  
34574
34575 /**
34576  * @class Roo.tree.DefaultSelectionModel
34577  * @extends Roo.util.Observable
34578  * The default single selection for a TreePanel.
34579  * @param {Object} cfg Configuration
34580  */
34581 Roo.tree.DefaultSelectionModel = function(cfg){
34582    this.selNode = null;
34583    
34584    
34585    
34586    this.addEvents({
34587        /**
34588         * @event selectionchange
34589         * Fires when the selected node changes
34590         * @param {DefaultSelectionModel} this
34591         * @param {TreeNode} node the new selection
34592         */
34593        "selectionchange" : true,
34594
34595        /**
34596         * @event beforeselect
34597         * Fires before the selected node changes, return false to cancel the change
34598         * @param {DefaultSelectionModel} this
34599         * @param {TreeNode} node the new selection
34600         * @param {TreeNode} node the old selection
34601         */
34602        "beforeselect" : true
34603    });
34604    
34605     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34606 };
34607
34608 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34609     init : function(tree){
34610         this.tree = tree;
34611         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34612         tree.on("click", this.onNodeClick, this);
34613     },
34614     
34615     onNodeClick : function(node, e){
34616         if (e.ctrlKey && this.selNode == node)  {
34617             this.unselect(node);
34618             return;
34619         }
34620         this.select(node);
34621     },
34622     
34623     /**
34624      * Select a node.
34625      * @param {TreeNode} node The node to select
34626      * @return {TreeNode} The selected node
34627      */
34628     select : function(node){
34629         var last = this.selNode;
34630         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34631             if(last){
34632                 last.ui.onSelectedChange(false);
34633             }
34634             this.selNode = node;
34635             node.ui.onSelectedChange(true);
34636             this.fireEvent("selectionchange", this, node, last);
34637         }
34638         return node;
34639     },
34640     
34641     /**
34642      * Deselect a node.
34643      * @param {TreeNode} node The node to unselect
34644      */
34645     unselect : function(node){
34646         if(this.selNode == node){
34647             this.clearSelections();
34648         }    
34649     },
34650     
34651     /**
34652      * Clear all selections
34653      */
34654     clearSelections : function(){
34655         var n = this.selNode;
34656         if(n){
34657             n.ui.onSelectedChange(false);
34658             this.selNode = null;
34659             this.fireEvent("selectionchange", this, null);
34660         }
34661         return n;
34662     },
34663     
34664     /**
34665      * Get the selected node
34666      * @return {TreeNode} The selected node
34667      */
34668     getSelectedNode : function(){
34669         return this.selNode;    
34670     },
34671     
34672     /**
34673      * Returns true if the node is selected
34674      * @param {TreeNode} node The node to check
34675      * @return {Boolean}
34676      */
34677     isSelected : function(node){
34678         return this.selNode == node;  
34679     },
34680
34681     /**
34682      * Selects the node above the selected node in the tree, intelligently walking the nodes
34683      * @return TreeNode The new selection
34684      */
34685     selectPrevious : function(){
34686         var s = this.selNode || this.lastSelNode;
34687         if(!s){
34688             return null;
34689         }
34690         var ps = s.previousSibling;
34691         if(ps){
34692             if(!ps.isExpanded() || ps.childNodes.length < 1){
34693                 return this.select(ps);
34694             } else{
34695                 var lc = ps.lastChild;
34696                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34697                     lc = lc.lastChild;
34698                 }
34699                 return this.select(lc);
34700             }
34701         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34702             return this.select(s.parentNode);
34703         }
34704         return null;
34705     },
34706
34707     /**
34708      * Selects the node above the selected node in the tree, intelligently walking the nodes
34709      * @return TreeNode The new selection
34710      */
34711     selectNext : function(){
34712         var s = this.selNode || this.lastSelNode;
34713         if(!s){
34714             return null;
34715         }
34716         if(s.firstChild && s.isExpanded()){
34717              return this.select(s.firstChild);
34718          }else if(s.nextSibling){
34719              return this.select(s.nextSibling);
34720          }else if(s.parentNode){
34721             var newS = null;
34722             s.parentNode.bubble(function(){
34723                 if(this.nextSibling){
34724                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34725                     return false;
34726                 }
34727             });
34728             return newS;
34729          }
34730         return null;
34731     },
34732
34733     onKeyDown : function(e){
34734         var s = this.selNode || this.lastSelNode;
34735         // undesirable, but required
34736         var sm = this;
34737         if(!s){
34738             return;
34739         }
34740         var k = e.getKey();
34741         switch(k){
34742              case e.DOWN:
34743                  e.stopEvent();
34744                  this.selectNext();
34745              break;
34746              case e.UP:
34747                  e.stopEvent();
34748                  this.selectPrevious();
34749              break;
34750              case e.RIGHT:
34751                  e.preventDefault();
34752                  if(s.hasChildNodes()){
34753                      if(!s.isExpanded()){
34754                          s.expand();
34755                      }else if(s.firstChild){
34756                          this.select(s.firstChild, e);
34757                      }
34758                  }
34759              break;
34760              case e.LEFT:
34761                  e.preventDefault();
34762                  if(s.hasChildNodes() && s.isExpanded()){
34763                      s.collapse();
34764                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34765                      this.select(s.parentNode, e);
34766                  }
34767              break;
34768         };
34769     }
34770 });
34771
34772 /**
34773  * @class Roo.tree.MultiSelectionModel
34774  * @extends Roo.util.Observable
34775  * Multi selection for a TreePanel.
34776  * @param {Object} cfg Configuration
34777  */
34778 Roo.tree.MultiSelectionModel = function(){
34779    this.selNodes = [];
34780    this.selMap = {};
34781    this.addEvents({
34782        /**
34783         * @event selectionchange
34784         * Fires when the selected nodes change
34785         * @param {MultiSelectionModel} this
34786         * @param {Array} nodes Array of the selected nodes
34787         */
34788        "selectionchange" : true
34789    });
34790    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34791    
34792 };
34793
34794 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34795     init : function(tree){
34796         this.tree = tree;
34797         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34798         tree.on("click", this.onNodeClick, this);
34799     },
34800     
34801     onNodeClick : function(node, e){
34802         this.select(node, e, e.ctrlKey);
34803     },
34804     
34805     /**
34806      * Select a node.
34807      * @param {TreeNode} node The node to select
34808      * @param {EventObject} e (optional) An event associated with the selection
34809      * @param {Boolean} keepExisting True to retain existing selections
34810      * @return {TreeNode} The selected node
34811      */
34812     select : function(node, e, keepExisting){
34813         if(keepExisting !== true){
34814             this.clearSelections(true);
34815         }
34816         if(this.isSelected(node)){
34817             this.lastSelNode = node;
34818             return node;
34819         }
34820         this.selNodes.push(node);
34821         this.selMap[node.id] = node;
34822         this.lastSelNode = node;
34823         node.ui.onSelectedChange(true);
34824         this.fireEvent("selectionchange", this, this.selNodes);
34825         return node;
34826     },
34827     
34828     /**
34829      * Deselect a node.
34830      * @param {TreeNode} node The node to unselect
34831      */
34832     unselect : function(node){
34833         if(this.selMap[node.id]){
34834             node.ui.onSelectedChange(false);
34835             var sn = this.selNodes;
34836             var index = -1;
34837             if(sn.indexOf){
34838                 index = sn.indexOf(node);
34839             }else{
34840                 for(var i = 0, len = sn.length; i < len; i++){
34841                     if(sn[i] == node){
34842                         index = i;
34843                         break;
34844                     }
34845                 }
34846             }
34847             if(index != -1){
34848                 this.selNodes.splice(index, 1);
34849             }
34850             delete this.selMap[node.id];
34851             this.fireEvent("selectionchange", this, this.selNodes);
34852         }
34853     },
34854     
34855     /**
34856      * Clear all selections
34857      */
34858     clearSelections : function(suppressEvent){
34859         var sn = this.selNodes;
34860         if(sn.length > 0){
34861             for(var i = 0, len = sn.length; i < len; i++){
34862                 sn[i].ui.onSelectedChange(false);
34863             }
34864             this.selNodes = [];
34865             this.selMap = {};
34866             if(suppressEvent !== true){
34867                 this.fireEvent("selectionchange", this, this.selNodes);
34868             }
34869         }
34870     },
34871     
34872     /**
34873      * Returns true if the node is selected
34874      * @param {TreeNode} node The node to check
34875      * @return {Boolean}
34876      */
34877     isSelected : function(node){
34878         return this.selMap[node.id] ? true : false;  
34879     },
34880     
34881     /**
34882      * Returns an array of the selected nodes
34883      * @return {Array}
34884      */
34885     getSelectedNodes : function(){
34886         return this.selNodes;    
34887     },
34888
34889     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34890
34891     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34892
34893     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34894 });/*
34895  * Based on:
34896  * Ext JS Library 1.1.1
34897  * Copyright(c) 2006-2007, Ext JS, LLC.
34898  *
34899  * Originally Released Under LGPL - original licence link has changed is not relivant.
34900  *
34901  * Fork - LGPL
34902  * <script type="text/javascript">
34903  */
34904  
34905 /**
34906  * @class Roo.tree.TreeNode
34907  * @extends Roo.data.Node
34908  * @cfg {String} text The text for this node
34909  * @cfg {Boolean} expanded true to start the node expanded
34910  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34911  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34912  * @cfg {Boolean} disabled true to start the node disabled
34913  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34914  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34915  * @cfg {String} cls A css class to be added to the node
34916  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34917  * @cfg {String} href URL of the link used for the node (defaults to #)
34918  * @cfg {String} hrefTarget target frame for the link
34919  * @cfg {String} qtip An Ext QuickTip for the node
34920  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34921  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34922  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34923  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34924  * (defaults to undefined with no checkbox rendered)
34925  * @constructor
34926  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34927  */
34928 Roo.tree.TreeNode = function(attributes){
34929     attributes = attributes || {};
34930     if(typeof attributes == "string"){
34931         attributes = {text: attributes};
34932     }
34933     this.childrenRendered = false;
34934     this.rendered = false;
34935     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34936     this.expanded = attributes.expanded === true;
34937     this.isTarget = attributes.isTarget !== false;
34938     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34939     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34940
34941     /**
34942      * Read-only. The text for this node. To change it use setText().
34943      * @type String
34944      */
34945     this.text = attributes.text;
34946     /**
34947      * True if this node is disabled.
34948      * @type Boolean
34949      */
34950     this.disabled = attributes.disabled === true;
34951
34952     this.addEvents({
34953         /**
34954         * @event textchange
34955         * Fires when the text for this node is changed
34956         * @param {Node} this This node
34957         * @param {String} text The new text
34958         * @param {String} oldText The old text
34959         */
34960         "textchange" : true,
34961         /**
34962         * @event beforeexpand
34963         * Fires before this node is expanded, return false to cancel.
34964         * @param {Node} this This node
34965         * @param {Boolean} deep
34966         * @param {Boolean} anim
34967         */
34968         "beforeexpand" : true,
34969         /**
34970         * @event beforecollapse
34971         * Fires before this node is collapsed, return false to cancel.
34972         * @param {Node} this This node
34973         * @param {Boolean} deep
34974         * @param {Boolean} anim
34975         */
34976         "beforecollapse" : true,
34977         /**
34978         * @event expand
34979         * Fires when this node is expanded
34980         * @param {Node} this This node
34981         */
34982         "expand" : true,
34983         /**
34984         * @event disabledchange
34985         * Fires when the disabled status of this node changes
34986         * @param {Node} this This node
34987         * @param {Boolean} disabled
34988         */
34989         "disabledchange" : true,
34990         /**
34991         * @event collapse
34992         * Fires when this node is collapsed
34993         * @param {Node} this This node
34994         */
34995         "collapse" : true,
34996         /**
34997         * @event beforeclick
34998         * Fires before click processing. Return false to cancel the default action.
34999         * @param {Node} this This node
35000         * @param {Roo.EventObject} e The event object
35001         */
35002         "beforeclick":true,
35003         /**
35004         * @event checkchange
35005         * Fires when a node with a checkbox's checked property changes
35006         * @param {Node} this This node
35007         * @param {Boolean} checked
35008         */
35009         "checkchange":true,
35010         /**
35011         * @event click
35012         * Fires when this node is clicked
35013         * @param {Node} this This node
35014         * @param {Roo.EventObject} e The event object
35015         */
35016         "click":true,
35017         /**
35018         * @event dblclick
35019         * Fires when this node is double clicked
35020         * @param {Node} this This node
35021         * @param {Roo.EventObject} e The event object
35022         */
35023         "dblclick":true,
35024         /**
35025         * @event contextmenu
35026         * Fires when this node is right clicked
35027         * @param {Node} this This node
35028         * @param {Roo.EventObject} e The event object
35029         */
35030         "contextmenu":true,
35031         /**
35032         * @event beforechildrenrendered
35033         * Fires right before the child nodes for this node are rendered
35034         * @param {Node} this This node
35035         */
35036         "beforechildrenrendered":true
35037     });
35038
35039     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35040
35041     /**
35042      * Read-only. The UI for this node
35043      * @type TreeNodeUI
35044      */
35045     this.ui = new uiClass(this);
35046     
35047     // finally support items[]
35048     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35049         return;
35050     }
35051     
35052     
35053     Roo.each(this.attributes.items, function(c) {
35054         this.appendChild(Roo.factory(c,Roo.Tree));
35055     }, this);
35056     delete this.attributes.items;
35057     
35058     
35059     
35060 };
35061 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35062     preventHScroll: true,
35063     /**
35064      * Returns true if this node is expanded
35065      * @return {Boolean}
35066      */
35067     isExpanded : function(){
35068         return this.expanded;
35069     },
35070
35071     /**
35072      * Returns the UI object for this node
35073      * @return {TreeNodeUI}
35074      */
35075     getUI : function(){
35076         return this.ui;
35077     },
35078
35079     // private override
35080     setFirstChild : function(node){
35081         var of = this.firstChild;
35082         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35083         if(this.childrenRendered && of && node != of){
35084             of.renderIndent(true, true);
35085         }
35086         if(this.rendered){
35087             this.renderIndent(true, true);
35088         }
35089     },
35090
35091     // private override
35092     setLastChild : function(node){
35093         var ol = this.lastChild;
35094         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35095         if(this.childrenRendered && ol && node != ol){
35096             ol.renderIndent(true, true);
35097         }
35098         if(this.rendered){
35099             this.renderIndent(true, true);
35100         }
35101     },
35102
35103     // these methods are overridden to provide lazy rendering support
35104     // private override
35105     appendChild : function()
35106     {
35107         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35108         if(node && this.childrenRendered){
35109             node.render();
35110         }
35111         this.ui.updateExpandIcon();
35112         return node;
35113     },
35114
35115     // private override
35116     removeChild : function(node){
35117         this.ownerTree.getSelectionModel().unselect(node);
35118         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35119         // if it's been rendered remove dom node
35120         if(this.childrenRendered){
35121             node.ui.remove();
35122         }
35123         if(this.childNodes.length < 1){
35124             this.collapse(false, false);
35125         }else{
35126             this.ui.updateExpandIcon();
35127         }
35128         if(!this.firstChild) {
35129             this.childrenRendered = false;
35130         }
35131         return node;
35132     },
35133
35134     // private override
35135     insertBefore : function(node, refNode){
35136         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35137         if(newNode && refNode && this.childrenRendered){
35138             node.render();
35139         }
35140         this.ui.updateExpandIcon();
35141         return newNode;
35142     },
35143
35144     /**
35145      * Sets the text for this node
35146      * @param {String} text
35147      */
35148     setText : function(text){
35149         var oldText = this.text;
35150         this.text = text;
35151         this.attributes.text = text;
35152         if(this.rendered){ // event without subscribing
35153             this.ui.onTextChange(this, text, oldText);
35154         }
35155         this.fireEvent("textchange", this, text, oldText);
35156     },
35157
35158     /**
35159      * Triggers selection of this node
35160      */
35161     select : function(){
35162         this.getOwnerTree().getSelectionModel().select(this);
35163     },
35164
35165     /**
35166      * Triggers deselection of this node
35167      */
35168     unselect : function(){
35169         this.getOwnerTree().getSelectionModel().unselect(this);
35170     },
35171
35172     /**
35173      * Returns true if this node is selected
35174      * @return {Boolean}
35175      */
35176     isSelected : function(){
35177         return this.getOwnerTree().getSelectionModel().isSelected(this);
35178     },
35179
35180     /**
35181      * Expand this node.
35182      * @param {Boolean} deep (optional) True to expand all children as well
35183      * @param {Boolean} anim (optional) false to cancel the default animation
35184      * @param {Function} callback (optional) A callback to be called when
35185      * expanding this node completes (does not wait for deep expand to complete).
35186      * Called with 1 parameter, this node.
35187      */
35188     expand : function(deep, anim, callback){
35189         if(!this.expanded){
35190             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35191                 return;
35192             }
35193             if(!this.childrenRendered){
35194                 this.renderChildren();
35195             }
35196             this.expanded = true;
35197             
35198             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35199                 this.ui.animExpand(function(){
35200                     this.fireEvent("expand", this);
35201                     if(typeof callback == "function"){
35202                         callback(this);
35203                     }
35204                     if(deep === true){
35205                         this.expandChildNodes(true);
35206                     }
35207                 }.createDelegate(this));
35208                 return;
35209             }else{
35210                 this.ui.expand();
35211                 this.fireEvent("expand", this);
35212                 if(typeof callback == "function"){
35213                     callback(this);
35214                 }
35215             }
35216         }else{
35217            if(typeof callback == "function"){
35218                callback(this);
35219            }
35220         }
35221         if(deep === true){
35222             this.expandChildNodes(true);
35223         }
35224     },
35225
35226     isHiddenRoot : function(){
35227         return this.isRoot && !this.getOwnerTree().rootVisible;
35228     },
35229
35230     /**
35231      * Collapse this node.
35232      * @param {Boolean} deep (optional) True to collapse all children as well
35233      * @param {Boolean} anim (optional) false to cancel the default animation
35234      */
35235     collapse : function(deep, anim){
35236         if(this.expanded && !this.isHiddenRoot()){
35237             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35238                 return;
35239             }
35240             this.expanded = false;
35241             if((this.getOwnerTree().animate && anim !== false) || anim){
35242                 this.ui.animCollapse(function(){
35243                     this.fireEvent("collapse", this);
35244                     if(deep === true){
35245                         this.collapseChildNodes(true);
35246                     }
35247                 }.createDelegate(this));
35248                 return;
35249             }else{
35250                 this.ui.collapse();
35251                 this.fireEvent("collapse", this);
35252             }
35253         }
35254         if(deep === true){
35255             var cs = this.childNodes;
35256             for(var i = 0, len = cs.length; i < len; i++) {
35257                 cs[i].collapse(true, false);
35258             }
35259         }
35260     },
35261
35262     // private
35263     delayedExpand : function(delay){
35264         if(!this.expandProcId){
35265             this.expandProcId = this.expand.defer(delay, this);
35266         }
35267     },
35268
35269     // private
35270     cancelExpand : function(){
35271         if(this.expandProcId){
35272             clearTimeout(this.expandProcId);
35273         }
35274         this.expandProcId = false;
35275     },
35276
35277     /**
35278      * Toggles expanded/collapsed state of the node
35279      */
35280     toggle : function(){
35281         if(this.expanded){
35282             this.collapse();
35283         }else{
35284             this.expand();
35285         }
35286     },
35287
35288     /**
35289      * Ensures all parent nodes are expanded
35290      */
35291     ensureVisible : function(callback){
35292         var tree = this.getOwnerTree();
35293         tree.expandPath(this.parentNode.getPath(), false, function(){
35294             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35295             Roo.callback(callback);
35296         }.createDelegate(this));
35297     },
35298
35299     /**
35300      * Expand all child nodes
35301      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35302      */
35303     expandChildNodes : function(deep){
35304         var cs = this.childNodes;
35305         for(var i = 0, len = cs.length; i < len; i++) {
35306                 cs[i].expand(deep);
35307         }
35308     },
35309
35310     /**
35311      * Collapse all child nodes
35312      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35313      */
35314     collapseChildNodes : function(deep){
35315         var cs = this.childNodes;
35316         for(var i = 0, len = cs.length; i < len; i++) {
35317                 cs[i].collapse(deep);
35318         }
35319     },
35320
35321     /**
35322      * Disables this node
35323      */
35324     disable : function(){
35325         this.disabled = true;
35326         this.unselect();
35327         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35328             this.ui.onDisableChange(this, true);
35329         }
35330         this.fireEvent("disabledchange", this, true);
35331     },
35332
35333     /**
35334      * Enables this node
35335      */
35336     enable : function(){
35337         this.disabled = false;
35338         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35339             this.ui.onDisableChange(this, false);
35340         }
35341         this.fireEvent("disabledchange", this, false);
35342     },
35343
35344     // private
35345     renderChildren : function(suppressEvent){
35346         if(suppressEvent !== false){
35347             this.fireEvent("beforechildrenrendered", this);
35348         }
35349         var cs = this.childNodes;
35350         for(var i = 0, len = cs.length; i < len; i++){
35351             cs[i].render(true);
35352         }
35353         this.childrenRendered = true;
35354     },
35355
35356     // private
35357     sort : function(fn, scope){
35358         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35359         if(this.childrenRendered){
35360             var cs = this.childNodes;
35361             for(var i = 0, len = cs.length; i < len; i++){
35362                 cs[i].render(true);
35363             }
35364         }
35365     },
35366
35367     // private
35368     render : function(bulkRender){
35369         this.ui.render(bulkRender);
35370         if(!this.rendered){
35371             this.rendered = true;
35372             if(this.expanded){
35373                 this.expanded = false;
35374                 this.expand(false, false);
35375             }
35376         }
35377     },
35378
35379     // private
35380     renderIndent : function(deep, refresh){
35381         if(refresh){
35382             this.ui.childIndent = null;
35383         }
35384         this.ui.renderIndent();
35385         if(deep === true && this.childrenRendered){
35386             var cs = this.childNodes;
35387             for(var i = 0, len = cs.length; i < len; i++){
35388                 cs[i].renderIndent(true, refresh);
35389             }
35390         }
35391     }
35392 });/*
35393  * Based on:
35394  * Ext JS Library 1.1.1
35395  * Copyright(c) 2006-2007, Ext JS, LLC.
35396  *
35397  * Originally Released Under LGPL - original licence link has changed is not relivant.
35398  *
35399  * Fork - LGPL
35400  * <script type="text/javascript">
35401  */
35402  
35403 /**
35404  * @class Roo.tree.AsyncTreeNode
35405  * @extends Roo.tree.TreeNode
35406  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35407  * @constructor
35408  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35409  */
35410  Roo.tree.AsyncTreeNode = function(config){
35411     this.loaded = false;
35412     this.loading = false;
35413     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35414     /**
35415     * @event beforeload
35416     * Fires before this node is loaded, return false to cancel
35417     * @param {Node} this This node
35418     */
35419     this.addEvents({'beforeload':true, 'load': true});
35420     /**
35421     * @event load
35422     * Fires when this node is loaded
35423     * @param {Node} this This node
35424     */
35425     /**
35426      * The loader used by this node (defaults to using the tree's defined loader)
35427      * @type TreeLoader
35428      * @property loader
35429      */
35430 };
35431 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35432     expand : function(deep, anim, callback){
35433         if(this.loading){ // if an async load is already running, waiting til it's done
35434             var timer;
35435             var f = function(){
35436                 if(!this.loading){ // done loading
35437                     clearInterval(timer);
35438                     this.expand(deep, anim, callback);
35439                 }
35440             }.createDelegate(this);
35441             timer = setInterval(f, 200);
35442             return;
35443         }
35444         if(!this.loaded){
35445             if(this.fireEvent("beforeload", this) === false){
35446                 return;
35447             }
35448             this.loading = true;
35449             this.ui.beforeLoad(this);
35450             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35451             if(loader){
35452                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35453                 return;
35454             }
35455         }
35456         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35457     },
35458     
35459     /**
35460      * Returns true if this node is currently loading
35461      * @return {Boolean}
35462      */
35463     isLoading : function(){
35464         return this.loading;  
35465     },
35466     
35467     loadComplete : function(deep, anim, callback){
35468         this.loading = false;
35469         this.loaded = true;
35470         this.ui.afterLoad(this);
35471         this.fireEvent("load", this);
35472         this.expand(deep, anim, callback);
35473     },
35474     
35475     /**
35476      * Returns true if this node has been loaded
35477      * @return {Boolean}
35478      */
35479     isLoaded : function(){
35480         return this.loaded;
35481     },
35482     
35483     hasChildNodes : function(){
35484         if(!this.isLeaf() && !this.loaded){
35485             return true;
35486         }else{
35487             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35488         }
35489     },
35490
35491     /**
35492      * Trigger a reload for this node
35493      * @param {Function} callback
35494      */
35495     reload : function(callback){
35496         this.collapse(false, false);
35497         while(this.firstChild){
35498             this.removeChild(this.firstChild);
35499         }
35500         this.childrenRendered = false;
35501         this.loaded = false;
35502         if(this.isHiddenRoot()){
35503             this.expanded = false;
35504         }
35505         this.expand(false, false, callback);
35506     }
35507 });/*
35508  * Based on:
35509  * Ext JS Library 1.1.1
35510  * Copyright(c) 2006-2007, Ext JS, LLC.
35511  *
35512  * Originally Released Under LGPL - original licence link has changed is not relivant.
35513  *
35514  * Fork - LGPL
35515  * <script type="text/javascript">
35516  */
35517  
35518 /**
35519  * @class Roo.tree.TreeNodeUI
35520  * @constructor
35521  * @param {Object} node The node to render
35522  * The TreeNode UI implementation is separate from the
35523  * tree implementation. Unless you are customizing the tree UI,
35524  * you should never have to use this directly.
35525  */
35526 Roo.tree.TreeNodeUI = function(node){
35527     this.node = node;
35528     this.rendered = false;
35529     this.animating = false;
35530     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35531 };
35532
35533 Roo.tree.TreeNodeUI.prototype = {
35534     removeChild : function(node){
35535         if(this.rendered){
35536             this.ctNode.removeChild(node.ui.getEl());
35537         }
35538     },
35539
35540     beforeLoad : function(){
35541          this.addClass("x-tree-node-loading");
35542     },
35543
35544     afterLoad : function(){
35545          this.removeClass("x-tree-node-loading");
35546     },
35547
35548     onTextChange : function(node, text, oldText){
35549         if(this.rendered){
35550             this.textNode.innerHTML = text;
35551         }
35552     },
35553
35554     onDisableChange : function(node, state){
35555         this.disabled = state;
35556         if(state){
35557             this.addClass("x-tree-node-disabled");
35558         }else{
35559             this.removeClass("x-tree-node-disabled");
35560         }
35561     },
35562
35563     onSelectedChange : function(state){
35564         if(state){
35565             this.focus();
35566             this.addClass("x-tree-selected");
35567         }else{
35568             //this.blur();
35569             this.removeClass("x-tree-selected");
35570         }
35571     },
35572
35573     onMove : function(tree, node, oldParent, newParent, index, refNode){
35574         this.childIndent = null;
35575         if(this.rendered){
35576             var targetNode = newParent.ui.getContainer();
35577             if(!targetNode){//target not rendered
35578                 this.holder = document.createElement("div");
35579                 this.holder.appendChild(this.wrap);
35580                 return;
35581             }
35582             var insertBefore = refNode ? refNode.ui.getEl() : null;
35583             if(insertBefore){
35584                 targetNode.insertBefore(this.wrap, insertBefore);
35585             }else{
35586                 targetNode.appendChild(this.wrap);
35587             }
35588             this.node.renderIndent(true);
35589         }
35590     },
35591
35592     addClass : function(cls){
35593         if(this.elNode){
35594             Roo.fly(this.elNode).addClass(cls);
35595         }
35596     },
35597
35598     removeClass : function(cls){
35599         if(this.elNode){
35600             Roo.fly(this.elNode).removeClass(cls);
35601         }
35602     },
35603
35604     remove : function(){
35605         if(this.rendered){
35606             this.holder = document.createElement("div");
35607             this.holder.appendChild(this.wrap);
35608         }
35609     },
35610
35611     fireEvent : function(){
35612         return this.node.fireEvent.apply(this.node, arguments);
35613     },
35614
35615     initEvents : function(){
35616         this.node.on("move", this.onMove, this);
35617         var E = Roo.EventManager;
35618         var a = this.anchor;
35619
35620         var el = Roo.fly(a, '_treeui');
35621
35622         if(Roo.isOpera){ // opera render bug ignores the CSS
35623             el.setStyle("text-decoration", "none");
35624         }
35625
35626         el.on("click", this.onClick, this);
35627         el.on("dblclick", this.onDblClick, this);
35628
35629         if(this.checkbox){
35630             Roo.EventManager.on(this.checkbox,
35631                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35632         }
35633
35634         el.on("contextmenu", this.onContextMenu, this);
35635
35636         var icon = Roo.fly(this.iconNode);
35637         icon.on("click", this.onClick, this);
35638         icon.on("dblclick", this.onDblClick, this);
35639         icon.on("contextmenu", this.onContextMenu, this);
35640         E.on(this.ecNode, "click", this.ecClick, this, true);
35641
35642         if(this.node.disabled){
35643             this.addClass("x-tree-node-disabled");
35644         }
35645         if(this.node.hidden){
35646             this.addClass("x-tree-node-disabled");
35647         }
35648         var ot = this.node.getOwnerTree();
35649         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35650         if(dd && (!this.node.isRoot || ot.rootVisible)){
35651             Roo.dd.Registry.register(this.elNode, {
35652                 node: this.node,
35653                 handles: this.getDDHandles(),
35654                 isHandle: false
35655             });
35656         }
35657     },
35658
35659     getDDHandles : function(){
35660         return [this.iconNode, this.textNode];
35661     },
35662
35663     hide : function(){
35664         if(this.rendered){
35665             this.wrap.style.display = "none";
35666         }
35667     },
35668
35669     show : function(){
35670         if(this.rendered){
35671             this.wrap.style.display = "";
35672         }
35673     },
35674
35675     onContextMenu : function(e){
35676         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35677             e.preventDefault();
35678             this.focus();
35679             this.fireEvent("contextmenu", this.node, e);
35680         }
35681     },
35682
35683     onClick : function(e){
35684         if(this.dropping){
35685             e.stopEvent();
35686             return;
35687         }
35688         if(this.fireEvent("beforeclick", this.node, e) !== false){
35689             if(!this.disabled && this.node.attributes.href){
35690                 this.fireEvent("click", this.node, e);
35691                 return;
35692             }
35693             e.preventDefault();
35694             if(this.disabled){
35695                 return;
35696             }
35697
35698             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35699                 this.node.toggle();
35700             }
35701
35702             this.fireEvent("click", this.node, e);
35703         }else{
35704             e.stopEvent();
35705         }
35706     },
35707
35708     onDblClick : function(e){
35709         e.preventDefault();
35710         if(this.disabled){
35711             return;
35712         }
35713         if(this.checkbox){
35714             this.toggleCheck();
35715         }
35716         if(!this.animating && this.node.hasChildNodes()){
35717             this.node.toggle();
35718         }
35719         this.fireEvent("dblclick", this.node, e);
35720     },
35721
35722     onCheckChange : function(){
35723         var checked = this.checkbox.checked;
35724         this.node.attributes.checked = checked;
35725         this.fireEvent('checkchange', this.node, checked);
35726     },
35727
35728     ecClick : function(e){
35729         if(!this.animating && this.node.hasChildNodes()){
35730             this.node.toggle();
35731         }
35732     },
35733
35734     startDrop : function(){
35735         this.dropping = true;
35736     },
35737
35738     // delayed drop so the click event doesn't get fired on a drop
35739     endDrop : function(){
35740        setTimeout(function(){
35741            this.dropping = false;
35742        }.createDelegate(this), 50);
35743     },
35744
35745     expand : function(){
35746         this.updateExpandIcon();
35747         this.ctNode.style.display = "";
35748     },
35749
35750     focus : function(){
35751         if(!this.node.preventHScroll){
35752             try{this.anchor.focus();
35753             }catch(e){}
35754         }else if(!Roo.isIE){
35755             try{
35756                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35757                 var l = noscroll.scrollLeft;
35758                 this.anchor.focus();
35759                 noscroll.scrollLeft = l;
35760             }catch(e){}
35761         }
35762     },
35763
35764     toggleCheck : function(value){
35765         var cb = this.checkbox;
35766         if(cb){
35767             cb.checked = (value === undefined ? !cb.checked : value);
35768         }
35769     },
35770
35771     blur : function(){
35772         try{
35773             this.anchor.blur();
35774         }catch(e){}
35775     },
35776
35777     animExpand : function(callback){
35778         var ct = Roo.get(this.ctNode);
35779         ct.stopFx();
35780         if(!this.node.hasChildNodes()){
35781             this.updateExpandIcon();
35782             this.ctNode.style.display = "";
35783             Roo.callback(callback);
35784             return;
35785         }
35786         this.animating = true;
35787         this.updateExpandIcon();
35788
35789         ct.slideIn('t', {
35790            callback : function(){
35791                this.animating = false;
35792                Roo.callback(callback);
35793             },
35794             scope: this,
35795             duration: this.node.ownerTree.duration || .25
35796         });
35797     },
35798
35799     highlight : function(){
35800         var tree = this.node.getOwnerTree();
35801         Roo.fly(this.wrap).highlight(
35802             tree.hlColor || "C3DAF9",
35803             {endColor: tree.hlBaseColor}
35804         );
35805     },
35806
35807     collapse : function(){
35808         this.updateExpandIcon();
35809         this.ctNode.style.display = "none";
35810     },
35811
35812     animCollapse : function(callback){
35813         var ct = Roo.get(this.ctNode);
35814         ct.enableDisplayMode('block');
35815         ct.stopFx();
35816
35817         this.animating = true;
35818         this.updateExpandIcon();
35819
35820         ct.slideOut('t', {
35821             callback : function(){
35822                this.animating = false;
35823                Roo.callback(callback);
35824             },
35825             scope: this,
35826             duration: this.node.ownerTree.duration || .25
35827         });
35828     },
35829
35830     getContainer : function(){
35831         return this.ctNode;
35832     },
35833
35834     getEl : function(){
35835         return this.wrap;
35836     },
35837
35838     appendDDGhost : function(ghostNode){
35839         ghostNode.appendChild(this.elNode.cloneNode(true));
35840     },
35841
35842     getDDRepairXY : function(){
35843         return Roo.lib.Dom.getXY(this.iconNode);
35844     },
35845
35846     onRender : function(){
35847         this.render();
35848     },
35849
35850     render : function(bulkRender){
35851         var n = this.node, a = n.attributes;
35852         var targetNode = n.parentNode ?
35853               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35854
35855         if(!this.rendered){
35856             this.rendered = true;
35857
35858             this.renderElements(n, a, targetNode, bulkRender);
35859
35860             if(a.qtip){
35861                if(this.textNode.setAttributeNS){
35862                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35863                    if(a.qtipTitle){
35864                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35865                    }
35866                }else{
35867                    this.textNode.setAttribute("ext:qtip", a.qtip);
35868                    if(a.qtipTitle){
35869                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35870                    }
35871                }
35872             }else if(a.qtipCfg){
35873                 a.qtipCfg.target = Roo.id(this.textNode);
35874                 Roo.QuickTips.register(a.qtipCfg);
35875             }
35876             this.initEvents();
35877             if(!this.node.expanded){
35878                 this.updateExpandIcon();
35879             }
35880         }else{
35881             if(bulkRender === true) {
35882                 targetNode.appendChild(this.wrap);
35883             }
35884         }
35885     },
35886
35887     renderElements : function(n, a, targetNode, bulkRender)
35888     {
35889         // add some indent caching, this helps performance when rendering a large tree
35890         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35891         var t = n.getOwnerTree();
35892         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35893         if (typeof(n.attributes.html) != 'undefined') {
35894             txt = n.attributes.html;
35895         }
35896         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35897         var cb = typeof a.checked == 'boolean';
35898         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35899         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35900             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35901             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35902             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35903             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35904             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35905              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35906                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35907             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35908             "</li>"];
35909
35910         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35911             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35912                                 n.nextSibling.ui.getEl(), buf.join(""));
35913         }else{
35914             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35915         }
35916
35917         this.elNode = this.wrap.childNodes[0];
35918         this.ctNode = this.wrap.childNodes[1];
35919         var cs = this.elNode.childNodes;
35920         this.indentNode = cs[0];
35921         this.ecNode = cs[1];
35922         this.iconNode = cs[2];
35923         var index = 3;
35924         if(cb){
35925             this.checkbox = cs[3];
35926             index++;
35927         }
35928         this.anchor = cs[index];
35929         this.textNode = cs[index].firstChild;
35930     },
35931
35932     getAnchor : function(){
35933         return this.anchor;
35934     },
35935
35936     getTextEl : function(){
35937         return this.textNode;
35938     },
35939
35940     getIconEl : function(){
35941         return this.iconNode;
35942     },
35943
35944     isChecked : function(){
35945         return this.checkbox ? this.checkbox.checked : false;
35946     },
35947
35948     updateExpandIcon : function(){
35949         if(this.rendered){
35950             var n = this.node, c1, c2;
35951             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35952             var hasChild = n.hasChildNodes();
35953             if(hasChild){
35954                 if(n.expanded){
35955                     cls += "-minus";
35956                     c1 = "x-tree-node-collapsed";
35957                     c2 = "x-tree-node-expanded";
35958                 }else{
35959                     cls += "-plus";
35960                     c1 = "x-tree-node-expanded";
35961                     c2 = "x-tree-node-collapsed";
35962                 }
35963                 if(this.wasLeaf){
35964                     this.removeClass("x-tree-node-leaf");
35965                     this.wasLeaf = false;
35966                 }
35967                 if(this.c1 != c1 || this.c2 != c2){
35968                     Roo.fly(this.elNode).replaceClass(c1, c2);
35969                     this.c1 = c1; this.c2 = c2;
35970                 }
35971             }else{
35972                 // this changes non-leafs into leafs if they have no children.
35973                 // it's not very rational behaviour..
35974                 
35975                 if(!this.wasLeaf && this.node.leaf){
35976                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35977                     delete this.c1;
35978                     delete this.c2;
35979                     this.wasLeaf = true;
35980                 }
35981             }
35982             var ecc = "x-tree-ec-icon "+cls;
35983             if(this.ecc != ecc){
35984                 this.ecNode.className = ecc;
35985                 this.ecc = ecc;
35986             }
35987         }
35988     },
35989
35990     getChildIndent : function(){
35991         if(!this.childIndent){
35992             var buf = [];
35993             var p = this.node;
35994             while(p){
35995                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35996                     if(!p.isLast()) {
35997                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35998                     } else {
35999                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
36000                     }
36001                 }
36002                 p = p.parentNode;
36003             }
36004             this.childIndent = buf.join("");
36005         }
36006         return this.childIndent;
36007     },
36008
36009     renderIndent : function(){
36010         if(this.rendered){
36011             var indent = "";
36012             var p = this.node.parentNode;
36013             if(p){
36014                 indent = p.ui.getChildIndent();
36015             }
36016             if(this.indentMarkup != indent){ // don't rerender if not required
36017                 this.indentNode.innerHTML = indent;
36018                 this.indentMarkup = indent;
36019             }
36020             this.updateExpandIcon();
36021         }
36022     }
36023 };
36024
36025 Roo.tree.RootTreeNodeUI = function(){
36026     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36027 };
36028 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36029     render : function(){
36030         if(!this.rendered){
36031             var targetNode = this.node.ownerTree.innerCt.dom;
36032             this.node.expanded = true;
36033             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36034             this.wrap = this.ctNode = targetNode.firstChild;
36035         }
36036     },
36037     collapse : function(){
36038     },
36039     expand : function(){
36040     }
36041 });/*
36042  * Based on:
36043  * Ext JS Library 1.1.1
36044  * Copyright(c) 2006-2007, Ext JS, LLC.
36045  *
36046  * Originally Released Under LGPL - original licence link has changed is not relivant.
36047  *
36048  * Fork - LGPL
36049  * <script type="text/javascript">
36050  */
36051 /**
36052  * @class Roo.tree.TreeLoader
36053  * @extends Roo.util.Observable
36054  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36055  * nodes from a specified URL. The response must be a javascript Array definition
36056  * who's elements are node definition objects. eg:
36057  * <pre><code>
36058 {  success : true,
36059    data :      [
36060    
36061     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36062     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36063     ]
36064 }
36065
36066
36067 </code></pre>
36068  * <br><br>
36069  * The old style respose with just an array is still supported, but not recommended.
36070  * <br><br>
36071  *
36072  * A server request is sent, and child nodes are loaded only when a node is expanded.
36073  * The loading node's id is passed to the server under the parameter name "node" to
36074  * enable the server to produce the correct child nodes.
36075  * <br><br>
36076  * To pass extra parameters, an event handler may be attached to the "beforeload"
36077  * event, and the parameters specified in the TreeLoader's baseParams property:
36078  * <pre><code>
36079     myTreeLoader.on("beforeload", function(treeLoader, node) {
36080         this.baseParams.category = node.attributes.category;
36081     }, this);
36082     
36083 </code></pre>
36084  *
36085  * This would pass an HTTP parameter called "category" to the server containing
36086  * the value of the Node's "category" attribute.
36087  * @constructor
36088  * Creates a new Treeloader.
36089  * @param {Object} config A config object containing config properties.
36090  */
36091 Roo.tree.TreeLoader = function(config){
36092     this.baseParams = {};
36093     this.requestMethod = "POST";
36094     Roo.apply(this, config);
36095
36096     this.addEvents({
36097     
36098         /**
36099          * @event beforeload
36100          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36101          * @param {Object} This TreeLoader object.
36102          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36103          * @param {Object} callback The callback function specified in the {@link #load} call.
36104          */
36105         beforeload : true,
36106         /**
36107          * @event load
36108          * Fires when the node has been successfuly loaded.
36109          * @param {Object} This TreeLoader object.
36110          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36111          * @param {Object} response The response object containing the data from the server.
36112          */
36113         load : true,
36114         /**
36115          * @event loadexception
36116          * Fires if the network request failed.
36117          * @param {Object} This TreeLoader object.
36118          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36119          * @param {Object} response The response object containing the data from the server.
36120          */
36121         loadexception : true,
36122         /**
36123          * @event create
36124          * Fires before a node is created, enabling you to return custom Node types 
36125          * @param {Object} This TreeLoader object.
36126          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36127          */
36128         create : true
36129     });
36130
36131     Roo.tree.TreeLoader.superclass.constructor.call(this);
36132 };
36133
36134 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36135     /**
36136     * @cfg {String} dataUrl The URL from which to request a Json string which
36137     * specifies an array of node definition object representing the child nodes
36138     * to be loaded.
36139     */
36140     /**
36141     * @cfg {String} requestMethod either GET or POST
36142     * defaults to POST (due to BC)
36143     * to be loaded.
36144     */
36145     /**
36146     * @cfg {Object} baseParams (optional) An object containing properties which
36147     * specify HTTP parameters to be passed to each request for child nodes.
36148     */
36149     /**
36150     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36151     * created by this loader. If the attributes sent by the server have an attribute in this object,
36152     * they take priority.
36153     */
36154     /**
36155     * @cfg {Object} uiProviders (optional) An object containing properties which
36156     * 
36157     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36158     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36159     * <i>uiProvider</i> attribute of a returned child node is a string rather
36160     * than a reference to a TreeNodeUI implementation, this that string value
36161     * is used as a property name in the uiProviders object. You can define the provider named
36162     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36163     */
36164     uiProviders : {},
36165
36166     /**
36167     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36168     * child nodes before loading.
36169     */
36170     clearOnLoad : true,
36171
36172     /**
36173     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36174     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36175     * Grid query { data : [ .....] }
36176     */
36177     
36178     root : false,
36179      /**
36180     * @cfg {String} queryParam (optional) 
36181     * Name of the query as it will be passed on the querystring (defaults to 'node')
36182     * eg. the request will be ?node=[id]
36183     */
36184     
36185     
36186     queryParam: false,
36187     
36188     /**
36189      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36190      * This is called automatically when a node is expanded, but may be used to reload
36191      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36192      * @param {Roo.tree.TreeNode} node
36193      * @param {Function} callback
36194      */
36195     load : function(node, callback){
36196         if(this.clearOnLoad){
36197             while(node.firstChild){
36198                 node.removeChild(node.firstChild);
36199             }
36200         }
36201         if(node.attributes.children){ // preloaded json children
36202             var cs = node.attributes.children;
36203             for(var i = 0, len = cs.length; i < len; i++){
36204                 node.appendChild(this.createNode(cs[i]));
36205             }
36206             if(typeof callback == "function"){
36207                 callback();
36208             }
36209         }else if(this.dataUrl){
36210             this.requestData(node, callback);
36211         }
36212     },
36213
36214     getParams: function(node){
36215         var buf = [], bp = this.baseParams;
36216         for(var key in bp){
36217             if(typeof bp[key] != "function"){
36218                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36219             }
36220         }
36221         var n = this.queryParam === false ? 'node' : this.queryParam;
36222         buf.push(n + "=", encodeURIComponent(node.id));
36223         return buf.join("");
36224     },
36225
36226     requestData : function(node, callback){
36227         if(this.fireEvent("beforeload", this, node, callback) !== false){
36228             this.transId = Roo.Ajax.request({
36229                 method:this.requestMethod,
36230                 url: this.dataUrl||this.url,
36231                 success: this.handleResponse,
36232                 failure: this.handleFailure,
36233                 scope: this,
36234                 argument: {callback: callback, node: node},
36235                 params: this.getParams(node)
36236             });
36237         }else{
36238             // if the load is cancelled, make sure we notify
36239             // the node that we are done
36240             if(typeof callback == "function"){
36241                 callback();
36242             }
36243         }
36244     },
36245
36246     isLoading : function(){
36247         return this.transId ? true : false;
36248     },
36249
36250     abort : function(){
36251         if(this.isLoading()){
36252             Roo.Ajax.abort(this.transId);
36253         }
36254     },
36255
36256     // private
36257     createNode : function(attr)
36258     {
36259         // apply baseAttrs, nice idea Corey!
36260         if(this.baseAttrs){
36261             Roo.applyIf(attr, this.baseAttrs);
36262         }
36263         if(this.applyLoader !== false){
36264             attr.loader = this;
36265         }
36266         // uiProvider = depreciated..
36267         
36268         if(typeof(attr.uiProvider) == 'string'){
36269            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36270                 /**  eval:var:attr */ eval(attr.uiProvider);
36271         }
36272         if(typeof(this.uiProviders['default']) != 'undefined') {
36273             attr.uiProvider = this.uiProviders['default'];
36274         }
36275         
36276         this.fireEvent('create', this, attr);
36277         
36278         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36279         return(attr.leaf ?
36280                         new Roo.tree.TreeNode(attr) :
36281                         new Roo.tree.AsyncTreeNode(attr));
36282     },
36283
36284     processResponse : function(response, node, callback)
36285     {
36286         var json = response.responseText;
36287         try {
36288             
36289             var o = Roo.decode(json);
36290             
36291             if (this.root === false && typeof(o.success) != undefined) {
36292                 this.root = 'data'; // the default behaviour for list like data..
36293                 }
36294                 
36295             if (this.root !== false &&  !o.success) {
36296                 // it's a failure condition.
36297                 var a = response.argument;
36298                 this.fireEvent("loadexception", this, a.node, response);
36299                 Roo.log("Load failed - should have a handler really");
36300                 return;
36301             }
36302             
36303             
36304             
36305             if (this.root !== false) {
36306                  o = o[this.root];
36307             }
36308             
36309             for(var i = 0, len = o.length; i < len; i++){
36310                 var n = this.createNode(o[i]);
36311                 if(n){
36312                     node.appendChild(n);
36313                 }
36314             }
36315             if(typeof callback == "function"){
36316                 callback(this, node);
36317             }
36318         }catch(e){
36319             this.handleFailure(response);
36320         }
36321     },
36322
36323     handleResponse : function(response){
36324         this.transId = false;
36325         var a = response.argument;
36326         this.processResponse(response, a.node, a.callback);
36327         this.fireEvent("load", this, a.node, response);
36328     },
36329
36330     handleFailure : function(response)
36331     {
36332         // should handle failure better..
36333         this.transId = false;
36334         var a = response.argument;
36335         this.fireEvent("loadexception", this, a.node, response);
36336         if(typeof a.callback == "function"){
36337             a.callback(this, a.node);
36338         }
36339     }
36340 });/*
36341  * Based on:
36342  * Ext JS Library 1.1.1
36343  * Copyright(c) 2006-2007, Ext JS, LLC.
36344  *
36345  * Originally Released Under LGPL - original licence link has changed is not relivant.
36346  *
36347  * Fork - LGPL
36348  * <script type="text/javascript">
36349  */
36350
36351 /**
36352 * @class Roo.tree.TreeFilter
36353 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36354 * @param {TreePanel} tree
36355 * @param {Object} config (optional)
36356  */
36357 Roo.tree.TreeFilter = function(tree, config){
36358     this.tree = tree;
36359     this.filtered = {};
36360     Roo.apply(this, config);
36361 };
36362
36363 Roo.tree.TreeFilter.prototype = {
36364     clearBlank:false,
36365     reverse:false,
36366     autoClear:false,
36367     remove:false,
36368
36369      /**
36370      * Filter the data by a specific attribute.
36371      * @param {String/RegExp} value Either string that the attribute value
36372      * should start with or a RegExp to test against the attribute
36373      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36374      * @param {TreeNode} startNode (optional) The node to start the filter at.
36375      */
36376     filter : function(value, attr, startNode){
36377         attr = attr || "text";
36378         var f;
36379         if(typeof value == "string"){
36380             var vlen = value.length;
36381             // auto clear empty filter
36382             if(vlen == 0 && this.clearBlank){
36383                 this.clear();
36384                 return;
36385             }
36386             value = value.toLowerCase();
36387             f = function(n){
36388                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36389             };
36390         }else if(value.exec){ // regex?
36391             f = function(n){
36392                 return value.test(n.attributes[attr]);
36393             };
36394         }else{
36395             throw 'Illegal filter type, must be string or regex';
36396         }
36397         this.filterBy(f, null, startNode);
36398         },
36399
36400     /**
36401      * Filter by a function. The passed function will be called with each
36402      * node in the tree (or from the startNode). If the function returns true, the node is kept
36403      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36404      * @param {Function} fn The filter function
36405      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36406      */
36407     filterBy : function(fn, scope, startNode){
36408         startNode = startNode || this.tree.root;
36409         if(this.autoClear){
36410             this.clear();
36411         }
36412         var af = this.filtered, rv = this.reverse;
36413         var f = function(n){
36414             if(n == startNode){
36415                 return true;
36416             }
36417             if(af[n.id]){
36418                 return false;
36419             }
36420             var m = fn.call(scope || n, n);
36421             if(!m || rv){
36422                 af[n.id] = n;
36423                 n.ui.hide();
36424                 return false;
36425             }
36426             return true;
36427         };
36428         startNode.cascade(f);
36429         if(this.remove){
36430            for(var id in af){
36431                if(typeof id != "function"){
36432                    var n = af[id];
36433                    if(n && n.parentNode){
36434                        n.parentNode.removeChild(n);
36435                    }
36436                }
36437            }
36438         }
36439     },
36440
36441     /**
36442      * Clears the current filter. Note: with the "remove" option
36443      * set a filter cannot be cleared.
36444      */
36445     clear : function(){
36446         var t = this.tree;
36447         var af = this.filtered;
36448         for(var id in af){
36449             if(typeof id != "function"){
36450                 var n = af[id];
36451                 if(n){
36452                     n.ui.show();
36453                 }
36454             }
36455         }
36456         this.filtered = {};
36457     }
36458 };
36459 /*
36460  * Based on:
36461  * Ext JS Library 1.1.1
36462  * Copyright(c) 2006-2007, Ext JS, LLC.
36463  *
36464  * Originally Released Under LGPL - original licence link has changed is not relivant.
36465  *
36466  * Fork - LGPL
36467  * <script type="text/javascript">
36468  */
36469  
36470
36471 /**
36472  * @class Roo.tree.TreeSorter
36473  * Provides sorting of nodes in a TreePanel
36474  * 
36475  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36476  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36477  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36478  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36479  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36480  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36481  * @constructor
36482  * @param {TreePanel} tree
36483  * @param {Object} config
36484  */
36485 Roo.tree.TreeSorter = function(tree, config){
36486     Roo.apply(this, config);
36487     tree.on("beforechildrenrendered", this.doSort, this);
36488     tree.on("append", this.updateSort, this);
36489     tree.on("insert", this.updateSort, this);
36490     
36491     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36492     var p = this.property || "text";
36493     var sortType = this.sortType;
36494     var fs = this.folderSort;
36495     var cs = this.caseSensitive === true;
36496     var leafAttr = this.leafAttr || 'leaf';
36497
36498     this.sortFn = function(n1, n2){
36499         if(fs){
36500             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36501                 return 1;
36502             }
36503             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36504                 return -1;
36505             }
36506         }
36507         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36508         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36509         if(v1 < v2){
36510                         return dsc ? +1 : -1;
36511                 }else if(v1 > v2){
36512                         return dsc ? -1 : +1;
36513         }else{
36514                 return 0;
36515         }
36516     };
36517 };
36518
36519 Roo.tree.TreeSorter.prototype = {
36520     doSort : function(node){
36521         node.sort(this.sortFn);
36522     },
36523     
36524     compareNodes : function(n1, n2){
36525         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36526     },
36527     
36528     updateSort : function(tree, node){
36529         if(node.childrenRendered){
36530             this.doSort.defer(1, this, [node]);
36531         }
36532     }
36533 };/*
36534  * Based on:
36535  * Ext JS Library 1.1.1
36536  * Copyright(c) 2006-2007, Ext JS, LLC.
36537  *
36538  * Originally Released Under LGPL - original licence link has changed is not relivant.
36539  *
36540  * Fork - LGPL
36541  * <script type="text/javascript">
36542  */
36543
36544 if(Roo.dd.DropZone){
36545     
36546 Roo.tree.TreeDropZone = function(tree, config){
36547     this.allowParentInsert = false;
36548     this.allowContainerDrop = false;
36549     this.appendOnly = false;
36550     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36551     this.tree = tree;
36552     this.lastInsertClass = "x-tree-no-status";
36553     this.dragOverData = {};
36554 };
36555
36556 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36557     ddGroup : "TreeDD",
36558     scroll:  true,
36559     
36560     expandDelay : 1000,
36561     
36562     expandNode : function(node){
36563         if(node.hasChildNodes() && !node.isExpanded()){
36564             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36565         }
36566     },
36567     
36568     queueExpand : function(node){
36569         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36570     },
36571     
36572     cancelExpand : function(){
36573         if(this.expandProcId){
36574             clearTimeout(this.expandProcId);
36575             this.expandProcId = false;
36576         }
36577     },
36578     
36579     isValidDropPoint : function(n, pt, dd, e, data){
36580         if(!n || !data){ return false; }
36581         var targetNode = n.node;
36582         var dropNode = data.node;
36583         // default drop rules
36584         if(!(targetNode && targetNode.isTarget && pt)){
36585             return false;
36586         }
36587         if(pt == "append" && targetNode.allowChildren === false){
36588             return false;
36589         }
36590         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36591             return false;
36592         }
36593         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36594             return false;
36595         }
36596         // reuse the object
36597         var overEvent = this.dragOverData;
36598         overEvent.tree = this.tree;
36599         overEvent.target = targetNode;
36600         overEvent.data = data;
36601         overEvent.point = pt;
36602         overEvent.source = dd;
36603         overEvent.rawEvent = e;
36604         overEvent.dropNode = dropNode;
36605         overEvent.cancel = false;  
36606         var result = this.tree.fireEvent("nodedragover", overEvent);
36607         return overEvent.cancel === false && result !== false;
36608     },
36609     
36610     getDropPoint : function(e, n, dd)
36611     {
36612         var tn = n.node;
36613         if(tn.isRoot){
36614             return tn.allowChildren !== false ? "append" : false; // always append for root
36615         }
36616         var dragEl = n.ddel;
36617         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36618         var y = Roo.lib.Event.getPageY(e);
36619         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36620         
36621         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36622         var noAppend = tn.allowChildren === false;
36623         if(this.appendOnly || tn.parentNode.allowChildren === false){
36624             return noAppend ? false : "append";
36625         }
36626         var noBelow = false;
36627         if(!this.allowParentInsert){
36628             noBelow = tn.hasChildNodes() && tn.isExpanded();
36629         }
36630         var q = (b - t) / (noAppend ? 2 : 3);
36631         if(y >= t && y < (t + q)){
36632             return "above";
36633         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36634             return "below";
36635         }else{
36636             return "append";
36637         }
36638     },
36639     
36640     onNodeEnter : function(n, dd, e, data)
36641     {
36642         this.cancelExpand();
36643     },
36644     
36645     onNodeOver : function(n, dd, e, data)
36646     {
36647        
36648         var pt = this.getDropPoint(e, n, dd);
36649         var node = n.node;
36650         
36651         // auto node expand check
36652         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36653             this.queueExpand(node);
36654         }else if(pt != "append"){
36655             this.cancelExpand();
36656         }
36657         
36658         // set the insert point style on the target node
36659         var returnCls = this.dropNotAllowed;
36660         if(this.isValidDropPoint(n, pt, dd, e, data)){
36661            if(pt){
36662                var el = n.ddel;
36663                var cls;
36664                if(pt == "above"){
36665                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36666                    cls = "x-tree-drag-insert-above";
36667                }else if(pt == "below"){
36668                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36669                    cls = "x-tree-drag-insert-below";
36670                }else{
36671                    returnCls = "x-tree-drop-ok-append";
36672                    cls = "x-tree-drag-append";
36673                }
36674                if(this.lastInsertClass != cls){
36675                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36676                    this.lastInsertClass = cls;
36677                }
36678            }
36679        }
36680        return returnCls;
36681     },
36682     
36683     onNodeOut : function(n, dd, e, data){
36684         
36685         this.cancelExpand();
36686         this.removeDropIndicators(n);
36687     },
36688     
36689     onNodeDrop : function(n, dd, e, data){
36690         var point = this.getDropPoint(e, n, dd);
36691         var targetNode = n.node;
36692         targetNode.ui.startDrop();
36693         if(!this.isValidDropPoint(n, point, dd, e, data)){
36694             targetNode.ui.endDrop();
36695             return false;
36696         }
36697         // first try to find the drop node
36698         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36699         var dropEvent = {
36700             tree : this.tree,
36701             target: targetNode,
36702             data: data,
36703             point: point,
36704             source: dd,
36705             rawEvent: e,
36706             dropNode: dropNode,
36707             cancel: !dropNode   
36708         };
36709         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36710         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36711             targetNode.ui.endDrop();
36712             return false;
36713         }
36714         // allow target changing
36715         targetNode = dropEvent.target;
36716         if(point == "append" && !targetNode.isExpanded()){
36717             targetNode.expand(false, null, function(){
36718                 this.completeDrop(dropEvent);
36719             }.createDelegate(this));
36720         }else{
36721             this.completeDrop(dropEvent);
36722         }
36723         return true;
36724     },
36725     
36726     completeDrop : function(de){
36727         var ns = de.dropNode, p = de.point, t = de.target;
36728         if(!(ns instanceof Array)){
36729             ns = [ns];
36730         }
36731         var n;
36732         for(var i = 0, len = ns.length; i < len; i++){
36733             n = ns[i];
36734             if(p == "above"){
36735                 t.parentNode.insertBefore(n, t);
36736             }else if(p == "below"){
36737                 t.parentNode.insertBefore(n, t.nextSibling);
36738             }else{
36739                 t.appendChild(n);
36740             }
36741         }
36742         n.ui.focus();
36743         if(this.tree.hlDrop){
36744             n.ui.highlight();
36745         }
36746         t.ui.endDrop();
36747         this.tree.fireEvent("nodedrop", de);
36748     },
36749     
36750     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36751         if(this.tree.hlDrop){
36752             dropNode.ui.focus();
36753             dropNode.ui.highlight();
36754         }
36755         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36756     },
36757     
36758     getTree : function(){
36759         return this.tree;
36760     },
36761     
36762     removeDropIndicators : function(n){
36763         if(n && n.ddel){
36764             var el = n.ddel;
36765             Roo.fly(el).removeClass([
36766                     "x-tree-drag-insert-above",
36767                     "x-tree-drag-insert-below",
36768                     "x-tree-drag-append"]);
36769             this.lastInsertClass = "_noclass";
36770         }
36771     },
36772     
36773     beforeDragDrop : function(target, e, id){
36774         this.cancelExpand();
36775         return true;
36776     },
36777     
36778     afterRepair : function(data){
36779         if(data && Roo.enableFx){
36780             data.node.ui.highlight();
36781         }
36782         this.hideProxy();
36783     } 
36784     
36785 });
36786
36787 }
36788 /*
36789  * Based on:
36790  * Ext JS Library 1.1.1
36791  * Copyright(c) 2006-2007, Ext JS, LLC.
36792  *
36793  * Originally Released Under LGPL - original licence link has changed is not relivant.
36794  *
36795  * Fork - LGPL
36796  * <script type="text/javascript">
36797  */
36798  
36799
36800 if(Roo.dd.DragZone){
36801 Roo.tree.TreeDragZone = function(tree, config){
36802     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36803     this.tree = tree;
36804 };
36805
36806 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36807     ddGroup : "TreeDD",
36808    
36809     onBeforeDrag : function(data, e){
36810         var n = data.node;
36811         return n && n.draggable && !n.disabled;
36812     },
36813      
36814     
36815     onInitDrag : function(e){
36816         var data = this.dragData;
36817         this.tree.getSelectionModel().select(data.node);
36818         this.proxy.update("");
36819         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36820         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36821     },
36822     
36823     getRepairXY : function(e, data){
36824         return data.node.ui.getDDRepairXY();
36825     },
36826     
36827     onEndDrag : function(data, e){
36828         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36829         
36830         
36831     },
36832     
36833     onValidDrop : function(dd, e, id){
36834         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36835         this.hideProxy();
36836     },
36837     
36838     beforeInvalidDrop : function(e, id){
36839         // this scrolls the original position back into view
36840         var sm = this.tree.getSelectionModel();
36841         sm.clearSelections();
36842         sm.select(this.dragData.node);
36843     }
36844 });
36845 }/*
36846  * Based on:
36847  * Ext JS Library 1.1.1
36848  * Copyright(c) 2006-2007, Ext JS, LLC.
36849  *
36850  * Originally Released Under LGPL - original licence link has changed is not relivant.
36851  *
36852  * Fork - LGPL
36853  * <script type="text/javascript">
36854  */
36855 /**
36856  * @class Roo.tree.TreeEditor
36857  * @extends Roo.Editor
36858  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36859  * as the editor field.
36860  * @constructor
36861  * @param {Object} config (used to be the tree panel.)
36862  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36863  * 
36864  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36865  * @cfg {Roo.form.TextField|Object} field The field configuration
36866  *
36867  * 
36868  */
36869 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36870     var tree = config;
36871     var field;
36872     if (oldconfig) { // old style..
36873         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36874     } else {
36875         // new style..
36876         tree = config.tree;
36877         config.field = config.field  || {};
36878         config.field.xtype = 'TextField';
36879         field = Roo.factory(config.field, Roo.form);
36880     }
36881     config = config || {};
36882     
36883     
36884     this.addEvents({
36885         /**
36886          * @event beforenodeedit
36887          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36888          * false from the handler of this event.
36889          * @param {Editor} this
36890          * @param {Roo.tree.Node} node 
36891          */
36892         "beforenodeedit" : true
36893     });
36894     
36895     //Roo.log(config);
36896     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36897
36898     this.tree = tree;
36899
36900     tree.on('beforeclick', this.beforeNodeClick, this);
36901     tree.getTreeEl().on('mousedown', this.hide, this);
36902     this.on('complete', this.updateNode, this);
36903     this.on('beforestartedit', this.fitToTree, this);
36904     this.on('startedit', this.bindScroll, this, {delay:10});
36905     this.on('specialkey', this.onSpecialKey, this);
36906 };
36907
36908 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36909     /**
36910      * @cfg {String} alignment
36911      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36912      */
36913     alignment: "l-l",
36914     // inherit
36915     autoSize: false,
36916     /**
36917      * @cfg {Boolean} hideEl
36918      * True to hide the bound element while the editor is displayed (defaults to false)
36919      */
36920     hideEl : false,
36921     /**
36922      * @cfg {String} cls
36923      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36924      */
36925     cls: "x-small-editor x-tree-editor",
36926     /**
36927      * @cfg {Boolean} shim
36928      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36929      */
36930     shim:false,
36931     // inherit
36932     shadow:"frame",
36933     /**
36934      * @cfg {Number} maxWidth
36935      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36936      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36937      * scroll and client offsets into account prior to each edit.
36938      */
36939     maxWidth: 250,
36940
36941     editDelay : 350,
36942
36943     // private
36944     fitToTree : function(ed, el){
36945         var td = this.tree.getTreeEl().dom, nd = el.dom;
36946         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36947             td.scrollLeft = nd.offsetLeft;
36948         }
36949         var w = Math.min(
36950                 this.maxWidth,
36951                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36952         this.setSize(w, '');
36953         
36954         return this.fireEvent('beforenodeedit', this, this.editNode);
36955         
36956     },
36957
36958     // private
36959     triggerEdit : function(node){
36960         this.completeEdit();
36961         this.editNode = node;
36962         this.startEdit(node.ui.textNode, node.text);
36963     },
36964
36965     // private
36966     bindScroll : function(){
36967         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36968     },
36969
36970     // private
36971     beforeNodeClick : function(node, e){
36972         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36973         this.lastClick = new Date();
36974         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36975             e.stopEvent();
36976             this.triggerEdit(node);
36977             return false;
36978         }
36979         return true;
36980     },
36981
36982     // private
36983     updateNode : function(ed, value){
36984         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36985         this.editNode.setText(value);
36986     },
36987
36988     // private
36989     onHide : function(){
36990         Roo.tree.TreeEditor.superclass.onHide.call(this);
36991         if(this.editNode){
36992             this.editNode.ui.focus();
36993         }
36994     },
36995
36996     // private
36997     onSpecialKey : function(field, e){
36998         var k = e.getKey();
36999         if(k == e.ESC){
37000             e.stopEvent();
37001             this.cancelEdit();
37002         }else if(k == e.ENTER && !e.hasModifier()){
37003             e.stopEvent();
37004             this.completeEdit();
37005         }
37006     }
37007 });//<Script type="text/javascript">
37008 /*
37009  * Based on:
37010  * Ext JS Library 1.1.1
37011  * Copyright(c) 2006-2007, Ext JS, LLC.
37012  *
37013  * Originally Released Under LGPL - original licence link has changed is not relivant.
37014  *
37015  * Fork - LGPL
37016  * <script type="text/javascript">
37017  */
37018  
37019 /**
37020  * Not documented??? - probably should be...
37021  */
37022
37023 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37024     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37025     
37026     renderElements : function(n, a, targetNode, bulkRender){
37027         //consel.log("renderElements?");
37028         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37029
37030         var t = n.getOwnerTree();
37031         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37032         
37033         var cols = t.columns;
37034         var bw = t.borderWidth;
37035         var c = cols[0];
37036         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37037          var cb = typeof a.checked == "boolean";
37038         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37039         var colcls = 'x-t-' + tid + '-c0';
37040         var buf = [
37041             '<li class="x-tree-node">',
37042             
37043                 
37044                 '<div class="x-tree-node-el ', a.cls,'">',
37045                     // extran...
37046                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37047                 
37048                 
37049                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37050                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37051                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37052                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37053                            (a.iconCls ? ' '+a.iconCls : ''),
37054                            '" unselectable="on" />',
37055                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37056                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37057                              
37058                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37059                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37060                             '<span unselectable="on" qtip="' + tx + '">',
37061                              tx,
37062                              '</span></a>' ,
37063                     '</div>',
37064                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37065                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37066                  ];
37067         for(var i = 1, len = cols.length; i < len; i++){
37068             c = cols[i];
37069             colcls = 'x-t-' + tid + '-c' +i;
37070             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37071             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37072                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37073                       "</div>");
37074          }
37075          
37076          buf.push(
37077             '</a>',
37078             '<div class="x-clear"></div></div>',
37079             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37080             "</li>");
37081         
37082         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37083             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37084                                 n.nextSibling.ui.getEl(), buf.join(""));
37085         }else{
37086             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37087         }
37088         var el = this.wrap.firstChild;
37089         this.elRow = el;
37090         this.elNode = el.firstChild;
37091         this.ranchor = el.childNodes[1];
37092         this.ctNode = this.wrap.childNodes[1];
37093         var cs = el.firstChild.childNodes;
37094         this.indentNode = cs[0];
37095         this.ecNode = cs[1];
37096         this.iconNode = cs[2];
37097         var index = 3;
37098         if(cb){
37099             this.checkbox = cs[3];
37100             index++;
37101         }
37102         this.anchor = cs[index];
37103         
37104         this.textNode = cs[index].firstChild;
37105         
37106         //el.on("click", this.onClick, this);
37107         //el.on("dblclick", this.onDblClick, this);
37108         
37109         
37110        // console.log(this);
37111     },
37112     initEvents : function(){
37113         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37114         
37115             
37116         var a = this.ranchor;
37117
37118         var el = Roo.get(a);
37119
37120         if(Roo.isOpera){ // opera render bug ignores the CSS
37121             el.setStyle("text-decoration", "none");
37122         }
37123
37124         el.on("click", this.onClick, this);
37125         el.on("dblclick", this.onDblClick, this);
37126         el.on("contextmenu", this.onContextMenu, this);
37127         
37128     },
37129     
37130     /*onSelectedChange : function(state){
37131         if(state){
37132             this.focus();
37133             this.addClass("x-tree-selected");
37134         }else{
37135             //this.blur();
37136             this.removeClass("x-tree-selected");
37137         }
37138     },*/
37139     addClass : function(cls){
37140         if(this.elRow){
37141             Roo.fly(this.elRow).addClass(cls);
37142         }
37143         
37144     },
37145     
37146     
37147     removeClass : function(cls){
37148         if(this.elRow){
37149             Roo.fly(this.elRow).removeClass(cls);
37150         }
37151     }
37152
37153     
37154     
37155 });//<Script type="text/javascript">
37156
37157 /*
37158  * Based on:
37159  * Ext JS Library 1.1.1
37160  * Copyright(c) 2006-2007, Ext JS, LLC.
37161  *
37162  * Originally Released Under LGPL - original licence link has changed is not relivant.
37163  *
37164  * Fork - LGPL
37165  * <script type="text/javascript">
37166  */
37167  
37168
37169 /**
37170  * @class Roo.tree.ColumnTree
37171  * @extends Roo.data.TreePanel
37172  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37173  * @cfg {int} borderWidth  compined right/left border allowance
37174  * @constructor
37175  * @param {String/HTMLElement/Element} el The container element
37176  * @param {Object} config
37177  */
37178 Roo.tree.ColumnTree =  function(el, config)
37179 {
37180    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37181    this.addEvents({
37182         /**
37183         * @event resize
37184         * Fire this event on a container when it resizes
37185         * @param {int} w Width
37186         * @param {int} h Height
37187         */
37188        "resize" : true
37189     });
37190     this.on('resize', this.onResize, this);
37191 };
37192
37193 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37194     //lines:false,
37195     
37196     
37197     borderWidth: Roo.isBorderBox ? 0 : 2, 
37198     headEls : false,
37199     
37200     render : function(){
37201         // add the header.....
37202        
37203         Roo.tree.ColumnTree.superclass.render.apply(this);
37204         
37205         this.el.addClass('x-column-tree');
37206         
37207         this.headers = this.el.createChild(
37208             {cls:'x-tree-headers'},this.innerCt.dom);
37209    
37210         var cols = this.columns, c;
37211         var totalWidth = 0;
37212         this.headEls = [];
37213         var  len = cols.length;
37214         for(var i = 0; i < len; i++){
37215              c = cols[i];
37216              totalWidth += c.width;
37217             this.headEls.push(this.headers.createChild({
37218                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37219                  cn: {
37220                      cls:'x-tree-hd-text',
37221                      html: c.header
37222                  },
37223                  style:'width:'+(c.width-this.borderWidth)+'px;'
37224              }));
37225         }
37226         this.headers.createChild({cls:'x-clear'});
37227         // prevent floats from wrapping when clipped
37228         this.headers.setWidth(totalWidth);
37229         //this.innerCt.setWidth(totalWidth);
37230         this.innerCt.setStyle({ overflow: 'auto' });
37231         this.onResize(this.width, this.height);
37232              
37233         
37234     },
37235     onResize : function(w,h)
37236     {
37237         this.height = h;
37238         this.width = w;
37239         // resize cols..
37240         this.innerCt.setWidth(this.width);
37241         this.innerCt.setHeight(this.height-20);
37242         
37243         // headers...
37244         var cols = this.columns, c;
37245         var totalWidth = 0;
37246         var expEl = false;
37247         var len = cols.length;
37248         for(var i = 0; i < len; i++){
37249             c = cols[i];
37250             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37251                 // it's the expander..
37252                 expEl  = this.headEls[i];
37253                 continue;
37254             }
37255             totalWidth += c.width;
37256             
37257         }
37258         if (expEl) {
37259             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37260         }
37261         this.headers.setWidth(w-20);
37262
37263         
37264         
37265         
37266     }
37267 });
37268 /*
37269  * Based on:
37270  * Ext JS Library 1.1.1
37271  * Copyright(c) 2006-2007, Ext JS, LLC.
37272  *
37273  * Originally Released Under LGPL - original licence link has changed is not relivant.
37274  *
37275  * Fork - LGPL
37276  * <script type="text/javascript">
37277  */
37278  
37279 /**
37280  * @class Roo.menu.Menu
37281  * @extends Roo.util.Observable
37282  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37283  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37284  * @constructor
37285  * Creates a new Menu
37286  * @param {Object} config Configuration options
37287  */
37288 Roo.menu.Menu = function(config){
37289     
37290     Roo.menu.Menu.superclass.constructor.call(this, config);
37291     
37292     this.id = this.id || Roo.id();
37293     this.addEvents({
37294         /**
37295          * @event beforeshow
37296          * Fires before this menu is displayed
37297          * @param {Roo.menu.Menu} this
37298          */
37299         beforeshow : true,
37300         /**
37301          * @event beforehide
37302          * Fires before this menu is hidden
37303          * @param {Roo.menu.Menu} this
37304          */
37305         beforehide : true,
37306         /**
37307          * @event show
37308          * Fires after this menu is displayed
37309          * @param {Roo.menu.Menu} this
37310          */
37311         show : true,
37312         /**
37313          * @event hide
37314          * Fires after this menu is hidden
37315          * @param {Roo.menu.Menu} this
37316          */
37317         hide : true,
37318         /**
37319          * @event click
37320          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37321          * @param {Roo.menu.Menu} this
37322          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37323          * @param {Roo.EventObject} e
37324          */
37325         click : true,
37326         /**
37327          * @event mouseover
37328          * Fires when the mouse is hovering over this menu
37329          * @param {Roo.menu.Menu} this
37330          * @param {Roo.EventObject} e
37331          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37332          */
37333         mouseover : true,
37334         /**
37335          * @event mouseout
37336          * Fires when the mouse exits this menu
37337          * @param {Roo.menu.Menu} this
37338          * @param {Roo.EventObject} e
37339          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37340          */
37341         mouseout : true,
37342         /**
37343          * @event itemclick
37344          * Fires when a menu item contained in this menu is clicked
37345          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37346          * @param {Roo.EventObject} e
37347          */
37348         itemclick: true
37349     });
37350     if (this.registerMenu) {
37351         Roo.menu.MenuMgr.register(this);
37352     }
37353     
37354     var mis = this.items;
37355     this.items = new Roo.util.MixedCollection();
37356     if(mis){
37357         this.add.apply(this, mis);
37358     }
37359 };
37360
37361 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37362     /**
37363      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37364      */
37365     minWidth : 120,
37366     /**
37367      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37368      * for bottom-right shadow (defaults to "sides")
37369      */
37370     shadow : "sides",
37371     /**
37372      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37373      * this menu (defaults to "tl-tr?")
37374      */
37375     subMenuAlign : "tl-tr?",
37376     /**
37377      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37378      * relative to its element of origin (defaults to "tl-bl?")
37379      */
37380     defaultAlign : "tl-bl?",
37381     /**
37382      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37383      */
37384     allowOtherMenus : false,
37385     /**
37386      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37387      */
37388     registerMenu : true,
37389
37390     hidden:true,
37391
37392     // private
37393     render : function(){
37394         if(this.el){
37395             return;
37396         }
37397         var el = this.el = new Roo.Layer({
37398             cls: "x-menu",
37399             shadow:this.shadow,
37400             constrain: false,
37401             parentEl: this.parentEl || document.body,
37402             zindex:15000
37403         });
37404
37405         this.keyNav = new Roo.menu.MenuNav(this);
37406
37407         if(this.plain){
37408             el.addClass("x-menu-plain");
37409         }
37410         if(this.cls){
37411             el.addClass(this.cls);
37412         }
37413         // generic focus element
37414         this.focusEl = el.createChild({
37415             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37416         });
37417         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37418         //disabling touch- as it's causing issues ..
37419         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37420         ul.on('click'   , this.onClick, this);
37421         
37422         
37423         ul.on("mouseover", this.onMouseOver, this);
37424         ul.on("mouseout", this.onMouseOut, this);
37425         this.items.each(function(item){
37426             if (item.hidden) {
37427                 return;
37428             }
37429             
37430             var li = document.createElement("li");
37431             li.className = "x-menu-list-item";
37432             ul.dom.appendChild(li);
37433             item.render(li, this);
37434         }, this);
37435         this.ul = ul;
37436         this.autoWidth();
37437     },
37438
37439     // private
37440     autoWidth : function(){
37441         var el = this.el, ul = this.ul;
37442         if(!el){
37443             return;
37444         }
37445         var w = this.width;
37446         if(w){
37447             el.setWidth(w);
37448         }else if(Roo.isIE){
37449             el.setWidth(this.minWidth);
37450             var t = el.dom.offsetWidth; // force recalc
37451             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37452         }
37453     },
37454
37455     // private
37456     delayAutoWidth : function(){
37457         if(this.rendered){
37458             if(!this.awTask){
37459                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37460             }
37461             this.awTask.delay(20);
37462         }
37463     },
37464
37465     // private
37466     findTargetItem : function(e){
37467         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37468         if(t && t.menuItemId){
37469             return this.items.get(t.menuItemId);
37470         }
37471     },
37472
37473     // private
37474     onClick : function(e){
37475         Roo.log("menu.onClick");
37476         var t = this.findTargetItem(e);
37477         if(!t){
37478             return;
37479         }
37480         Roo.log(e);
37481         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37482             if(t == this.activeItem && t.shouldDeactivate(e)){
37483                 this.activeItem.deactivate();
37484                 delete this.activeItem;
37485                 return;
37486             }
37487             if(t.canActivate){
37488                 this.setActiveItem(t, true);
37489             }
37490             return;
37491             
37492             
37493         }
37494         
37495         t.onClick(e);
37496         this.fireEvent("click", this, t, e);
37497     },
37498
37499     // private
37500     setActiveItem : function(item, autoExpand){
37501         if(item != this.activeItem){
37502             if(this.activeItem){
37503                 this.activeItem.deactivate();
37504             }
37505             this.activeItem = item;
37506             item.activate(autoExpand);
37507         }else if(autoExpand){
37508             item.expandMenu();
37509         }
37510     },
37511
37512     // private
37513     tryActivate : function(start, step){
37514         var items = this.items;
37515         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37516             var item = items.get(i);
37517             if(!item.disabled && item.canActivate){
37518                 this.setActiveItem(item, false);
37519                 return item;
37520             }
37521         }
37522         return false;
37523     },
37524
37525     // private
37526     onMouseOver : function(e){
37527         var t;
37528         if(t = this.findTargetItem(e)){
37529             if(t.canActivate && !t.disabled){
37530                 this.setActiveItem(t, true);
37531             }
37532         }
37533         this.fireEvent("mouseover", this, e, t);
37534     },
37535
37536     // private
37537     onMouseOut : function(e){
37538         var t;
37539         if(t = this.findTargetItem(e)){
37540             if(t == this.activeItem && t.shouldDeactivate(e)){
37541                 this.activeItem.deactivate();
37542                 delete this.activeItem;
37543             }
37544         }
37545         this.fireEvent("mouseout", this, e, t);
37546     },
37547
37548     /**
37549      * Read-only.  Returns true if the menu is currently displayed, else false.
37550      * @type Boolean
37551      */
37552     isVisible : function(){
37553         return this.el && !this.hidden;
37554     },
37555
37556     /**
37557      * Displays this menu relative to another element
37558      * @param {String/HTMLElement/Roo.Element} element The element to align to
37559      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37560      * the element (defaults to this.defaultAlign)
37561      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37562      */
37563     show : function(el, pos, parentMenu){
37564         this.parentMenu = parentMenu;
37565         if(!this.el){
37566             this.render();
37567         }
37568         this.fireEvent("beforeshow", this);
37569         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37570     },
37571
37572     /**
37573      * Displays this menu at a specific xy position
37574      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37575      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37576      */
37577     showAt : function(xy, parentMenu, /* private: */_e){
37578         this.parentMenu = parentMenu;
37579         if(!this.el){
37580             this.render();
37581         }
37582         if(_e !== false){
37583             this.fireEvent("beforeshow", this);
37584             xy = this.el.adjustForConstraints(xy);
37585         }
37586         this.el.setXY(xy);
37587         this.el.show();
37588         this.hidden = false;
37589         this.focus();
37590         this.fireEvent("show", this);
37591     },
37592
37593     focus : function(){
37594         if(!this.hidden){
37595             this.doFocus.defer(50, this);
37596         }
37597     },
37598
37599     doFocus : function(){
37600         if(!this.hidden){
37601             this.focusEl.focus();
37602         }
37603     },
37604
37605     /**
37606      * Hides this menu and optionally all parent menus
37607      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37608      */
37609     hide : function(deep){
37610         if(this.el && this.isVisible()){
37611             this.fireEvent("beforehide", this);
37612             if(this.activeItem){
37613                 this.activeItem.deactivate();
37614                 this.activeItem = null;
37615             }
37616             this.el.hide();
37617             this.hidden = true;
37618             this.fireEvent("hide", this);
37619         }
37620         if(deep === true && this.parentMenu){
37621             this.parentMenu.hide(true);
37622         }
37623     },
37624
37625     /**
37626      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37627      * Any of the following are valid:
37628      * <ul>
37629      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37630      * <li>An HTMLElement object which will be converted to a menu item</li>
37631      * <li>A menu item config object that will be created as a new menu item</li>
37632      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37633      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37634      * </ul>
37635      * Usage:
37636      * <pre><code>
37637 // Create the menu
37638 var menu = new Roo.menu.Menu();
37639
37640 // Create a menu item to add by reference
37641 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37642
37643 // Add a bunch of items at once using different methods.
37644 // Only the last item added will be returned.
37645 var item = menu.add(
37646     menuItem,                // add existing item by ref
37647     'Dynamic Item',          // new TextItem
37648     '-',                     // new separator
37649     { text: 'Config Item' }  // new item by config
37650 );
37651 </code></pre>
37652      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37653      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37654      */
37655     add : function(){
37656         var a = arguments, l = a.length, item;
37657         for(var i = 0; i < l; i++){
37658             var el = a[i];
37659             if ((typeof(el) == "object") && el.xtype && el.xns) {
37660                 el = Roo.factory(el, Roo.menu);
37661             }
37662             
37663             if(el.render){ // some kind of Item
37664                 item = this.addItem(el);
37665             }else if(typeof el == "string"){ // string
37666                 if(el == "separator" || el == "-"){
37667                     item = this.addSeparator();
37668                 }else{
37669                     item = this.addText(el);
37670                 }
37671             }else if(el.tagName || el.el){ // element
37672                 item = this.addElement(el);
37673             }else if(typeof el == "object"){ // must be menu item config?
37674                 item = this.addMenuItem(el);
37675             }
37676         }
37677         return item;
37678     },
37679
37680     /**
37681      * Returns this menu's underlying {@link Roo.Element} object
37682      * @return {Roo.Element} The element
37683      */
37684     getEl : function(){
37685         if(!this.el){
37686             this.render();
37687         }
37688         return this.el;
37689     },
37690
37691     /**
37692      * Adds a separator bar to the menu
37693      * @return {Roo.menu.Item} The menu item that was added
37694      */
37695     addSeparator : function(){
37696         return this.addItem(new Roo.menu.Separator());
37697     },
37698
37699     /**
37700      * Adds an {@link Roo.Element} object to the menu
37701      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37702      * @return {Roo.menu.Item} The menu item that was added
37703      */
37704     addElement : function(el){
37705         return this.addItem(new Roo.menu.BaseItem(el));
37706     },
37707
37708     /**
37709      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37710      * @param {Roo.menu.Item} item The menu item to add
37711      * @return {Roo.menu.Item} The menu item that was added
37712      */
37713     addItem : function(item){
37714         this.items.add(item);
37715         if(this.ul){
37716             var li = document.createElement("li");
37717             li.className = "x-menu-list-item";
37718             this.ul.dom.appendChild(li);
37719             item.render(li, this);
37720             this.delayAutoWidth();
37721         }
37722         return item;
37723     },
37724
37725     /**
37726      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37727      * @param {Object} config A MenuItem config object
37728      * @return {Roo.menu.Item} The menu item that was added
37729      */
37730     addMenuItem : function(config){
37731         if(!(config instanceof Roo.menu.Item)){
37732             if(typeof config.checked == "boolean"){ // must be check menu item config?
37733                 config = new Roo.menu.CheckItem(config);
37734             }else{
37735                 config = new Roo.menu.Item(config);
37736             }
37737         }
37738         return this.addItem(config);
37739     },
37740
37741     /**
37742      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37743      * @param {String} text The text to display in the menu item
37744      * @return {Roo.menu.Item} The menu item that was added
37745      */
37746     addText : function(text){
37747         return this.addItem(new Roo.menu.TextItem({ text : text }));
37748     },
37749
37750     /**
37751      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37752      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37753      * @param {Roo.menu.Item} item The menu item to add
37754      * @return {Roo.menu.Item} The menu item that was added
37755      */
37756     insert : function(index, item){
37757         this.items.insert(index, item);
37758         if(this.ul){
37759             var li = document.createElement("li");
37760             li.className = "x-menu-list-item";
37761             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37762             item.render(li, this);
37763             this.delayAutoWidth();
37764         }
37765         return item;
37766     },
37767
37768     /**
37769      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37770      * @param {Roo.menu.Item} item The menu item to remove
37771      */
37772     remove : function(item){
37773         this.items.removeKey(item.id);
37774         item.destroy();
37775     },
37776
37777     /**
37778      * Removes and destroys all items in the menu
37779      */
37780     removeAll : function(){
37781         var f;
37782         while(f = this.items.first()){
37783             this.remove(f);
37784         }
37785     }
37786 });
37787
37788 // MenuNav is a private utility class used internally by the Menu
37789 Roo.menu.MenuNav = function(menu){
37790     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37791     this.scope = this.menu = menu;
37792 };
37793
37794 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37795     doRelay : function(e, h){
37796         var k = e.getKey();
37797         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37798             this.menu.tryActivate(0, 1);
37799             return false;
37800         }
37801         return h.call(this.scope || this, e, this.menu);
37802     },
37803
37804     up : function(e, m){
37805         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37806             m.tryActivate(m.items.length-1, -1);
37807         }
37808     },
37809
37810     down : function(e, m){
37811         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37812             m.tryActivate(0, 1);
37813         }
37814     },
37815
37816     right : function(e, m){
37817         if(m.activeItem){
37818             m.activeItem.expandMenu(true);
37819         }
37820     },
37821
37822     left : function(e, m){
37823         m.hide();
37824         if(m.parentMenu && m.parentMenu.activeItem){
37825             m.parentMenu.activeItem.activate();
37826         }
37827     },
37828
37829     enter : function(e, m){
37830         if(m.activeItem){
37831             e.stopPropagation();
37832             m.activeItem.onClick(e);
37833             m.fireEvent("click", this, m.activeItem);
37834             return true;
37835         }
37836     }
37837 });/*
37838  * Based on:
37839  * Ext JS Library 1.1.1
37840  * Copyright(c) 2006-2007, Ext JS, LLC.
37841  *
37842  * Originally Released Under LGPL - original licence link has changed is not relivant.
37843  *
37844  * Fork - LGPL
37845  * <script type="text/javascript">
37846  */
37847  
37848 /**
37849  * @class Roo.menu.MenuMgr
37850  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37851  * @singleton
37852  */
37853 Roo.menu.MenuMgr = function(){
37854    var menus, active, groups = {}, attached = false, lastShow = new Date();
37855
37856    // private - called when first menu is created
37857    function init(){
37858        menus = {};
37859        active = new Roo.util.MixedCollection();
37860        Roo.get(document).addKeyListener(27, function(){
37861            if(active.length > 0){
37862                hideAll();
37863            }
37864        });
37865    }
37866
37867    // private
37868    function hideAll(){
37869        if(active && active.length > 0){
37870            var c = active.clone();
37871            c.each(function(m){
37872                m.hide();
37873            });
37874        }
37875    }
37876
37877    // private
37878    function onHide(m){
37879        active.remove(m);
37880        if(active.length < 1){
37881            Roo.get(document).un("mousedown", onMouseDown);
37882            attached = false;
37883        }
37884    }
37885
37886    // private
37887    function onShow(m){
37888        var last = active.last();
37889        lastShow = new Date();
37890        active.add(m);
37891        if(!attached){
37892            Roo.get(document).on("mousedown", onMouseDown);
37893            attached = true;
37894        }
37895        if(m.parentMenu){
37896           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37897           m.parentMenu.activeChild = m;
37898        }else if(last && last.isVisible()){
37899           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37900        }
37901    }
37902
37903    // private
37904    function onBeforeHide(m){
37905        if(m.activeChild){
37906            m.activeChild.hide();
37907        }
37908        if(m.autoHideTimer){
37909            clearTimeout(m.autoHideTimer);
37910            delete m.autoHideTimer;
37911        }
37912    }
37913
37914    // private
37915    function onBeforeShow(m){
37916        var pm = m.parentMenu;
37917        if(!pm && !m.allowOtherMenus){
37918            hideAll();
37919        }else if(pm && pm.activeChild && active != m){
37920            pm.activeChild.hide();
37921        }
37922    }
37923
37924    // private
37925    function onMouseDown(e){
37926        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37927            hideAll();
37928        }
37929    }
37930
37931    // private
37932    function onBeforeCheck(mi, state){
37933        if(state){
37934            var g = groups[mi.group];
37935            for(var i = 0, l = g.length; i < l; i++){
37936                if(g[i] != mi){
37937                    g[i].setChecked(false);
37938                }
37939            }
37940        }
37941    }
37942
37943    return {
37944
37945        /**
37946         * Hides all menus that are currently visible
37947         */
37948        hideAll : function(){
37949             hideAll();  
37950        },
37951
37952        // private
37953        register : function(menu){
37954            if(!menus){
37955                init();
37956            }
37957            menus[menu.id] = menu;
37958            menu.on("beforehide", onBeforeHide);
37959            menu.on("hide", onHide);
37960            menu.on("beforeshow", onBeforeShow);
37961            menu.on("show", onShow);
37962            var g = menu.group;
37963            if(g && menu.events["checkchange"]){
37964                if(!groups[g]){
37965                    groups[g] = [];
37966                }
37967                groups[g].push(menu);
37968                menu.on("checkchange", onCheck);
37969            }
37970        },
37971
37972         /**
37973          * Returns a {@link Roo.menu.Menu} object
37974          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37975          * be used to generate and return a new Menu instance.
37976          */
37977        get : function(menu){
37978            if(typeof menu == "string"){ // menu id
37979                return menus[menu];
37980            }else if(menu.events){  // menu instance
37981                return menu;
37982            }else if(typeof menu.length == 'number'){ // array of menu items?
37983                return new Roo.menu.Menu({items:menu});
37984            }else{ // otherwise, must be a config
37985                return new Roo.menu.Menu(menu);
37986            }
37987        },
37988
37989        // private
37990        unregister : function(menu){
37991            delete menus[menu.id];
37992            menu.un("beforehide", onBeforeHide);
37993            menu.un("hide", onHide);
37994            menu.un("beforeshow", onBeforeShow);
37995            menu.un("show", onShow);
37996            var g = menu.group;
37997            if(g && menu.events["checkchange"]){
37998                groups[g].remove(menu);
37999                menu.un("checkchange", onCheck);
38000            }
38001        },
38002
38003        // private
38004        registerCheckable : function(menuItem){
38005            var g = menuItem.group;
38006            if(g){
38007                if(!groups[g]){
38008                    groups[g] = [];
38009                }
38010                groups[g].push(menuItem);
38011                menuItem.on("beforecheckchange", onBeforeCheck);
38012            }
38013        },
38014
38015        // private
38016        unregisterCheckable : function(menuItem){
38017            var g = menuItem.group;
38018            if(g){
38019                groups[g].remove(menuItem);
38020                menuItem.un("beforecheckchange", onBeforeCheck);
38021            }
38022        }
38023    };
38024 }();/*
38025  * Based on:
38026  * Ext JS Library 1.1.1
38027  * Copyright(c) 2006-2007, Ext JS, LLC.
38028  *
38029  * Originally Released Under LGPL - original licence link has changed is not relivant.
38030  *
38031  * Fork - LGPL
38032  * <script type="text/javascript">
38033  */
38034  
38035
38036 /**
38037  * @class Roo.menu.BaseItem
38038  * @extends Roo.Component
38039  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38040  * management and base configuration options shared by all menu components.
38041  * @constructor
38042  * Creates a new BaseItem
38043  * @param {Object} config Configuration options
38044  */
38045 Roo.menu.BaseItem = function(config){
38046     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38047
38048     this.addEvents({
38049         /**
38050          * @event click
38051          * Fires when this item is clicked
38052          * @param {Roo.menu.BaseItem} this
38053          * @param {Roo.EventObject} e
38054          */
38055         click: true,
38056         /**
38057          * @event activate
38058          * Fires when this item is activated
38059          * @param {Roo.menu.BaseItem} this
38060          */
38061         activate : true,
38062         /**
38063          * @event deactivate
38064          * Fires when this item is deactivated
38065          * @param {Roo.menu.BaseItem} this
38066          */
38067         deactivate : true
38068     });
38069
38070     if(this.handler){
38071         this.on("click", this.handler, this.scope, true);
38072     }
38073 };
38074
38075 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38076     /**
38077      * @cfg {Function} handler
38078      * A function that will handle the click event of this menu item (defaults to undefined)
38079      */
38080     /**
38081      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38082      */
38083     canActivate : false,
38084     
38085      /**
38086      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38087      */
38088     hidden: false,
38089     
38090     /**
38091      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38092      */
38093     activeClass : "x-menu-item-active",
38094     /**
38095      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38096      */
38097     hideOnClick : true,
38098     /**
38099      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38100      */
38101     hideDelay : 100,
38102
38103     // private
38104     ctype: "Roo.menu.BaseItem",
38105
38106     // private
38107     actionMode : "container",
38108
38109     // private
38110     render : function(container, parentMenu){
38111         this.parentMenu = parentMenu;
38112         Roo.menu.BaseItem.superclass.render.call(this, container);
38113         this.container.menuItemId = this.id;
38114     },
38115
38116     // private
38117     onRender : function(container, position){
38118         this.el = Roo.get(this.el);
38119         container.dom.appendChild(this.el.dom);
38120     },
38121
38122     // private
38123     onClick : function(e){
38124         if(!this.disabled && this.fireEvent("click", this, e) !== false
38125                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38126             this.handleClick(e);
38127         }else{
38128             e.stopEvent();
38129         }
38130     },
38131
38132     // private
38133     activate : function(){
38134         if(this.disabled){
38135             return false;
38136         }
38137         var li = this.container;
38138         li.addClass(this.activeClass);
38139         this.region = li.getRegion().adjust(2, 2, -2, -2);
38140         this.fireEvent("activate", this);
38141         return true;
38142     },
38143
38144     // private
38145     deactivate : function(){
38146         this.container.removeClass(this.activeClass);
38147         this.fireEvent("deactivate", this);
38148     },
38149
38150     // private
38151     shouldDeactivate : function(e){
38152         return !this.region || !this.region.contains(e.getPoint());
38153     },
38154
38155     // private
38156     handleClick : function(e){
38157         if(this.hideOnClick){
38158             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38159         }
38160     },
38161
38162     // private
38163     expandMenu : function(autoActivate){
38164         // do nothing
38165     },
38166
38167     // private
38168     hideMenu : function(){
38169         // do nothing
38170     }
38171 });/*
38172  * Based on:
38173  * Ext JS Library 1.1.1
38174  * Copyright(c) 2006-2007, Ext JS, LLC.
38175  *
38176  * Originally Released Under LGPL - original licence link has changed is not relivant.
38177  *
38178  * Fork - LGPL
38179  * <script type="text/javascript">
38180  */
38181  
38182 /**
38183  * @class Roo.menu.Adapter
38184  * @extends Roo.menu.BaseItem
38185  * 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.
38186  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38187  * @constructor
38188  * Creates a new Adapter
38189  * @param {Object} config Configuration options
38190  */
38191 Roo.menu.Adapter = function(component, config){
38192     Roo.menu.Adapter.superclass.constructor.call(this, config);
38193     this.component = component;
38194 };
38195 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38196     // private
38197     canActivate : true,
38198
38199     // private
38200     onRender : function(container, position){
38201         this.component.render(container);
38202         this.el = this.component.getEl();
38203     },
38204
38205     // private
38206     activate : function(){
38207         if(this.disabled){
38208             return false;
38209         }
38210         this.component.focus();
38211         this.fireEvent("activate", this);
38212         return true;
38213     },
38214
38215     // private
38216     deactivate : function(){
38217         this.fireEvent("deactivate", this);
38218     },
38219
38220     // private
38221     disable : function(){
38222         this.component.disable();
38223         Roo.menu.Adapter.superclass.disable.call(this);
38224     },
38225
38226     // private
38227     enable : function(){
38228         this.component.enable();
38229         Roo.menu.Adapter.superclass.enable.call(this);
38230     }
38231 });/*
38232  * Based on:
38233  * Ext JS Library 1.1.1
38234  * Copyright(c) 2006-2007, Ext JS, LLC.
38235  *
38236  * Originally Released Under LGPL - original licence link has changed is not relivant.
38237  *
38238  * Fork - LGPL
38239  * <script type="text/javascript">
38240  */
38241
38242 /**
38243  * @class Roo.menu.TextItem
38244  * @extends Roo.menu.BaseItem
38245  * Adds a static text string to a menu, usually used as either a heading or group separator.
38246  * Note: old style constructor with text is still supported.
38247  * 
38248  * @constructor
38249  * Creates a new TextItem
38250  * @param {Object} cfg Configuration
38251  */
38252 Roo.menu.TextItem = function(cfg){
38253     if (typeof(cfg) == 'string') {
38254         this.text = cfg;
38255     } else {
38256         Roo.apply(this,cfg);
38257     }
38258     
38259     Roo.menu.TextItem.superclass.constructor.call(this);
38260 };
38261
38262 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38263     /**
38264      * @cfg {Boolean} text Text to show on item.
38265      */
38266     text : '',
38267     
38268     /**
38269      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38270      */
38271     hideOnClick : false,
38272     /**
38273      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38274      */
38275     itemCls : "x-menu-text",
38276
38277     // private
38278     onRender : function(){
38279         var s = document.createElement("span");
38280         s.className = this.itemCls;
38281         s.innerHTML = this.text;
38282         this.el = s;
38283         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38284     }
38285 });/*
38286  * Based on:
38287  * Ext JS Library 1.1.1
38288  * Copyright(c) 2006-2007, Ext JS, LLC.
38289  *
38290  * Originally Released Under LGPL - original licence link has changed is not relivant.
38291  *
38292  * Fork - LGPL
38293  * <script type="text/javascript">
38294  */
38295
38296 /**
38297  * @class Roo.menu.Separator
38298  * @extends Roo.menu.BaseItem
38299  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38300  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38301  * @constructor
38302  * @param {Object} config Configuration options
38303  */
38304 Roo.menu.Separator = function(config){
38305     Roo.menu.Separator.superclass.constructor.call(this, config);
38306 };
38307
38308 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38309     /**
38310      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38311      */
38312     itemCls : "x-menu-sep",
38313     /**
38314      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38315      */
38316     hideOnClick : false,
38317
38318     // private
38319     onRender : function(li){
38320         var s = document.createElement("span");
38321         s.className = this.itemCls;
38322         s.innerHTML = "&#160;";
38323         this.el = s;
38324         li.addClass("x-menu-sep-li");
38325         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38326     }
38327 });/*
38328  * Based on:
38329  * Ext JS Library 1.1.1
38330  * Copyright(c) 2006-2007, Ext JS, LLC.
38331  *
38332  * Originally Released Under LGPL - original licence link has changed is not relivant.
38333  *
38334  * Fork - LGPL
38335  * <script type="text/javascript">
38336  */
38337 /**
38338  * @class Roo.menu.Item
38339  * @extends Roo.menu.BaseItem
38340  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38341  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38342  * activation and click handling.
38343  * @constructor
38344  * Creates a new Item
38345  * @param {Object} config Configuration options
38346  */
38347 Roo.menu.Item = function(config){
38348     Roo.menu.Item.superclass.constructor.call(this, config);
38349     if(this.menu){
38350         this.menu = Roo.menu.MenuMgr.get(this.menu);
38351     }
38352 };
38353 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38354     
38355     /**
38356      * @cfg {String} text
38357      * The text to show on the menu item.
38358      */
38359     text: '',
38360      /**
38361      * @cfg {String} HTML to render in menu
38362      * The text to show on the menu item (HTML version).
38363      */
38364     html: '',
38365     /**
38366      * @cfg {String} icon
38367      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38368      */
38369     icon: undefined,
38370     /**
38371      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38372      */
38373     itemCls : "x-menu-item",
38374     /**
38375      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38376      */
38377     canActivate : true,
38378     /**
38379      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38380      */
38381     showDelay: 200,
38382     // doc'd in BaseItem
38383     hideDelay: 200,
38384
38385     // private
38386     ctype: "Roo.menu.Item",
38387     
38388     // private
38389     onRender : function(container, position){
38390         var el = document.createElement("a");
38391         el.hideFocus = true;
38392         el.unselectable = "on";
38393         el.href = this.href || "#";
38394         if(this.hrefTarget){
38395             el.target = this.hrefTarget;
38396         }
38397         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38398         
38399         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38400         
38401         el.innerHTML = String.format(
38402                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38403                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38404         this.el = el;
38405         Roo.menu.Item.superclass.onRender.call(this, container, position);
38406     },
38407
38408     /**
38409      * Sets the text to display in this menu item
38410      * @param {String} text The text to display
38411      * @param {Boolean} isHTML true to indicate text is pure html.
38412      */
38413     setText : function(text, isHTML){
38414         if (isHTML) {
38415             this.html = text;
38416         } else {
38417             this.text = text;
38418             this.html = '';
38419         }
38420         if(this.rendered){
38421             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38422      
38423             this.el.update(String.format(
38424                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38425                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38426             this.parentMenu.autoWidth();
38427         }
38428     },
38429
38430     // private
38431     handleClick : function(e){
38432         if(!this.href){ // if no link defined, stop the event automatically
38433             e.stopEvent();
38434         }
38435         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38436     },
38437
38438     // private
38439     activate : function(autoExpand){
38440         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38441             this.focus();
38442             if(autoExpand){
38443                 this.expandMenu();
38444             }
38445         }
38446         return true;
38447     },
38448
38449     // private
38450     shouldDeactivate : function(e){
38451         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38452             if(this.menu && this.menu.isVisible()){
38453                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38454             }
38455             return true;
38456         }
38457         return false;
38458     },
38459
38460     // private
38461     deactivate : function(){
38462         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38463         this.hideMenu();
38464     },
38465
38466     // private
38467     expandMenu : function(autoActivate){
38468         if(!this.disabled && this.menu){
38469             clearTimeout(this.hideTimer);
38470             delete this.hideTimer;
38471             if(!this.menu.isVisible() && !this.showTimer){
38472                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38473             }else if (this.menu.isVisible() && autoActivate){
38474                 this.menu.tryActivate(0, 1);
38475             }
38476         }
38477     },
38478
38479     // private
38480     deferExpand : function(autoActivate){
38481         delete this.showTimer;
38482         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38483         if(autoActivate){
38484             this.menu.tryActivate(0, 1);
38485         }
38486     },
38487
38488     // private
38489     hideMenu : function(){
38490         clearTimeout(this.showTimer);
38491         delete this.showTimer;
38492         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38493             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38494         }
38495     },
38496
38497     // private
38498     deferHide : function(){
38499         delete this.hideTimer;
38500         this.menu.hide();
38501     }
38502 });/*
38503  * Based on:
38504  * Ext JS Library 1.1.1
38505  * Copyright(c) 2006-2007, Ext JS, LLC.
38506  *
38507  * Originally Released Under LGPL - original licence link has changed is not relivant.
38508  *
38509  * Fork - LGPL
38510  * <script type="text/javascript">
38511  */
38512  
38513 /**
38514  * @class Roo.menu.CheckItem
38515  * @extends Roo.menu.Item
38516  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38517  * @constructor
38518  * Creates a new CheckItem
38519  * @param {Object} config Configuration options
38520  */
38521 Roo.menu.CheckItem = function(config){
38522     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38523     this.addEvents({
38524         /**
38525          * @event beforecheckchange
38526          * Fires before the checked value is set, providing an opportunity to cancel if needed
38527          * @param {Roo.menu.CheckItem} this
38528          * @param {Boolean} checked The new checked value that will be set
38529          */
38530         "beforecheckchange" : true,
38531         /**
38532          * @event checkchange
38533          * Fires after the checked value has been set
38534          * @param {Roo.menu.CheckItem} this
38535          * @param {Boolean} checked The checked value that was set
38536          */
38537         "checkchange" : true
38538     });
38539     if(this.checkHandler){
38540         this.on('checkchange', this.checkHandler, this.scope);
38541     }
38542 };
38543 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38544     /**
38545      * @cfg {String} group
38546      * All check items with the same group name will automatically be grouped into a single-select
38547      * radio button group (defaults to '')
38548      */
38549     /**
38550      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38551      */
38552     itemCls : "x-menu-item x-menu-check-item",
38553     /**
38554      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38555      */
38556     groupClass : "x-menu-group-item",
38557
38558     /**
38559      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38560      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38561      * initialized with checked = true will be rendered as checked.
38562      */
38563     checked: false,
38564
38565     // private
38566     ctype: "Roo.menu.CheckItem",
38567
38568     // private
38569     onRender : function(c){
38570         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38571         if(this.group){
38572             this.el.addClass(this.groupClass);
38573         }
38574         Roo.menu.MenuMgr.registerCheckable(this);
38575         if(this.checked){
38576             this.checked = false;
38577             this.setChecked(true, true);
38578         }
38579     },
38580
38581     // private
38582     destroy : function(){
38583         if(this.rendered){
38584             Roo.menu.MenuMgr.unregisterCheckable(this);
38585         }
38586         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38587     },
38588
38589     /**
38590      * Set the checked state of this item
38591      * @param {Boolean} checked The new checked value
38592      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38593      */
38594     setChecked : function(state, suppressEvent){
38595         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38596             if(this.container){
38597                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38598             }
38599             this.checked = state;
38600             if(suppressEvent !== true){
38601                 this.fireEvent("checkchange", this, state);
38602             }
38603         }
38604     },
38605
38606     // private
38607     handleClick : function(e){
38608        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38609            this.setChecked(!this.checked);
38610        }
38611        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38612     }
38613 });/*
38614  * Based on:
38615  * Ext JS Library 1.1.1
38616  * Copyright(c) 2006-2007, Ext JS, LLC.
38617  *
38618  * Originally Released Under LGPL - original licence link has changed is not relivant.
38619  *
38620  * Fork - LGPL
38621  * <script type="text/javascript">
38622  */
38623  
38624 /**
38625  * @class Roo.menu.DateItem
38626  * @extends Roo.menu.Adapter
38627  * A menu item that wraps the {@link Roo.DatPicker} component.
38628  * @constructor
38629  * Creates a new DateItem
38630  * @param {Object} config Configuration options
38631  */
38632 Roo.menu.DateItem = function(config){
38633     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38634     /** The Roo.DatePicker object @type Roo.DatePicker */
38635     this.picker = this.component;
38636     this.addEvents({select: true});
38637     
38638     this.picker.on("render", function(picker){
38639         picker.getEl().swallowEvent("click");
38640         picker.container.addClass("x-menu-date-item");
38641     });
38642
38643     this.picker.on("select", this.onSelect, this);
38644 };
38645
38646 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38647     // private
38648     onSelect : function(picker, date){
38649         this.fireEvent("select", this, date, picker);
38650         Roo.menu.DateItem.superclass.handleClick.call(this);
38651     }
38652 });/*
38653  * Based on:
38654  * Ext JS Library 1.1.1
38655  * Copyright(c) 2006-2007, Ext JS, LLC.
38656  *
38657  * Originally Released Under LGPL - original licence link has changed is not relivant.
38658  *
38659  * Fork - LGPL
38660  * <script type="text/javascript">
38661  */
38662  
38663 /**
38664  * @class Roo.menu.ColorItem
38665  * @extends Roo.menu.Adapter
38666  * A menu item that wraps the {@link Roo.ColorPalette} component.
38667  * @constructor
38668  * Creates a new ColorItem
38669  * @param {Object} config Configuration options
38670  */
38671 Roo.menu.ColorItem = function(config){
38672     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38673     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38674     this.palette = this.component;
38675     this.relayEvents(this.palette, ["select"]);
38676     if(this.selectHandler){
38677         this.on('select', this.selectHandler, this.scope);
38678     }
38679 };
38680 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38681  * Based on:
38682  * Ext JS Library 1.1.1
38683  * Copyright(c) 2006-2007, Ext JS, LLC.
38684  *
38685  * Originally Released Under LGPL - original licence link has changed is not relivant.
38686  *
38687  * Fork - LGPL
38688  * <script type="text/javascript">
38689  */
38690  
38691
38692 /**
38693  * @class Roo.menu.DateMenu
38694  * @extends Roo.menu.Menu
38695  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38696  * @constructor
38697  * Creates a new DateMenu
38698  * @param {Object} config Configuration options
38699  */
38700 Roo.menu.DateMenu = function(config){
38701     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38702     this.plain = true;
38703     var di = new Roo.menu.DateItem(config);
38704     this.add(di);
38705     /**
38706      * The {@link Roo.DatePicker} instance for this DateMenu
38707      * @type DatePicker
38708      */
38709     this.picker = di.picker;
38710     /**
38711      * @event select
38712      * @param {DatePicker} picker
38713      * @param {Date} date
38714      */
38715     this.relayEvents(di, ["select"]);
38716     this.on('beforeshow', function(){
38717         if(this.picker){
38718             this.picker.hideMonthPicker(false);
38719         }
38720     }, this);
38721 };
38722 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38723     cls:'x-date-menu'
38724 });/*
38725  * Based on:
38726  * Ext JS Library 1.1.1
38727  * Copyright(c) 2006-2007, Ext JS, LLC.
38728  *
38729  * Originally Released Under LGPL - original licence link has changed is not relivant.
38730  *
38731  * Fork - LGPL
38732  * <script type="text/javascript">
38733  */
38734  
38735
38736 /**
38737  * @class Roo.menu.ColorMenu
38738  * @extends Roo.menu.Menu
38739  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38740  * @constructor
38741  * Creates a new ColorMenu
38742  * @param {Object} config Configuration options
38743  */
38744 Roo.menu.ColorMenu = function(config){
38745     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38746     this.plain = true;
38747     var ci = new Roo.menu.ColorItem(config);
38748     this.add(ci);
38749     /**
38750      * The {@link Roo.ColorPalette} instance for this ColorMenu
38751      * @type ColorPalette
38752      */
38753     this.palette = ci.palette;
38754     /**
38755      * @event select
38756      * @param {ColorPalette} palette
38757      * @param {String} color
38758      */
38759     this.relayEvents(ci, ["select"]);
38760 };
38761 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38762  * Based on:
38763  * Ext JS Library 1.1.1
38764  * Copyright(c) 2006-2007, Ext JS, LLC.
38765  *
38766  * Originally Released Under LGPL - original licence link has changed is not relivant.
38767  *
38768  * Fork - LGPL
38769  * <script type="text/javascript">
38770  */
38771  
38772 /**
38773  * @class Roo.form.TextItem
38774  * @extends Roo.BoxComponent
38775  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38776  * @constructor
38777  * Creates a new TextItem
38778  * @param {Object} config Configuration options
38779  */
38780 Roo.form.TextItem = function(config){
38781     Roo.form.TextItem.superclass.constructor.call(this, config);
38782 };
38783
38784 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38785     
38786     /**
38787      * @cfg {String} tag the tag for this item (default div)
38788      */
38789     tag : 'div',
38790     /**
38791      * @cfg {String} html the content for this item
38792      */
38793     html : '',
38794     
38795     getAutoCreate : function()
38796     {
38797         var cfg = {
38798             id: this.id,
38799             tag: this.tag,
38800             html: this.html,
38801             cls: 'x-form-item'
38802         };
38803         
38804         return cfg;
38805         
38806     },
38807     
38808     onRender : function(ct, position)
38809     {
38810         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38811         
38812         if(!this.el){
38813             var cfg = this.getAutoCreate();
38814             if(!cfg.name){
38815                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38816             }
38817             if (!cfg.name.length) {
38818                 delete cfg.name;
38819             }
38820             this.el = ct.createChild(cfg, position);
38821         }
38822     }
38823     
38824 });/*
38825  * Based on:
38826  * Ext JS Library 1.1.1
38827  * Copyright(c) 2006-2007, Ext JS, LLC.
38828  *
38829  * Originally Released Under LGPL - original licence link has changed is not relivant.
38830  *
38831  * Fork - LGPL
38832  * <script type="text/javascript">
38833  */
38834  
38835 /**
38836  * @class Roo.form.Field
38837  * @extends Roo.BoxComponent
38838  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38839  * @constructor
38840  * Creates a new Field
38841  * @param {Object} config Configuration options
38842  */
38843 Roo.form.Field = function(config){
38844     Roo.form.Field.superclass.constructor.call(this, config);
38845 };
38846
38847 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38848     /**
38849      * @cfg {String} fieldLabel Label to use when rendering a form.
38850      */
38851        /**
38852      * @cfg {String} qtip Mouse over tip
38853      */
38854      
38855     /**
38856      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38857      */
38858     invalidClass : "x-form-invalid",
38859     /**
38860      * @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")
38861      */
38862     invalidText : "The value in this field is invalid",
38863     /**
38864      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38865      */
38866     focusClass : "x-form-focus",
38867     /**
38868      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38869       automatic validation (defaults to "keyup").
38870      */
38871     validationEvent : "keyup",
38872     /**
38873      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38874      */
38875     validateOnBlur : true,
38876     /**
38877      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38878      */
38879     validationDelay : 250,
38880     /**
38881      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38882      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38883      */
38884     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38885     /**
38886      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38887      */
38888     fieldClass : "x-form-field",
38889     /**
38890      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38891      *<pre>
38892 Value         Description
38893 -----------   ----------------------------------------------------------------------
38894 qtip          Display a quick tip when the user hovers over the field
38895 title         Display a default browser title attribute popup
38896 under         Add a block div beneath the field containing the error text
38897 side          Add an error icon to the right of the field with a popup on hover
38898 [element id]  Add the error text directly to the innerHTML of the specified element
38899 </pre>
38900      */
38901     msgTarget : 'qtip',
38902     /**
38903      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38904      */
38905     msgFx : 'normal',
38906
38907     /**
38908      * @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.
38909      */
38910     readOnly : false,
38911
38912     /**
38913      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38914      */
38915     disabled : false,
38916
38917     /**
38918      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38919      */
38920     inputType : undefined,
38921     
38922     /**
38923      * @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).
38924          */
38925         tabIndex : undefined,
38926         
38927     // private
38928     isFormField : true,
38929
38930     // private
38931     hasFocus : false,
38932     /**
38933      * @property {Roo.Element} fieldEl
38934      * Element Containing the rendered Field (with label etc.)
38935      */
38936     /**
38937      * @cfg {Mixed} value A value to initialize this field with.
38938      */
38939     value : undefined,
38940
38941     /**
38942      * @cfg {String} name The field's HTML name attribute.
38943      */
38944     /**
38945      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38946      */
38947     // private
38948     loadedValue : false,
38949      
38950      
38951         // private ??
38952         initComponent : function(){
38953         Roo.form.Field.superclass.initComponent.call(this);
38954         this.addEvents({
38955             /**
38956              * @event focus
38957              * Fires when this field receives input focus.
38958              * @param {Roo.form.Field} this
38959              */
38960             focus : true,
38961             /**
38962              * @event blur
38963              * Fires when this field loses input focus.
38964              * @param {Roo.form.Field} this
38965              */
38966             blur : true,
38967             /**
38968              * @event specialkey
38969              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38970              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38971              * @param {Roo.form.Field} this
38972              * @param {Roo.EventObject} e The event object
38973              */
38974             specialkey : true,
38975             /**
38976              * @event change
38977              * Fires just before the field blurs if the field value has changed.
38978              * @param {Roo.form.Field} this
38979              * @param {Mixed} newValue The new value
38980              * @param {Mixed} oldValue The original value
38981              */
38982             change : true,
38983             /**
38984              * @event invalid
38985              * Fires after the field has been marked as invalid.
38986              * @param {Roo.form.Field} this
38987              * @param {String} msg The validation message
38988              */
38989             invalid : true,
38990             /**
38991              * @event valid
38992              * Fires after the field has been validated with no errors.
38993              * @param {Roo.form.Field} this
38994              */
38995             valid : true,
38996              /**
38997              * @event keyup
38998              * Fires after the key up
38999              * @param {Roo.form.Field} this
39000              * @param {Roo.EventObject}  e The event Object
39001              */
39002             keyup : true
39003         });
39004     },
39005
39006     /**
39007      * Returns the name attribute of the field if available
39008      * @return {String} name The field name
39009      */
39010     getName: function(){
39011          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39012     },
39013
39014     // private
39015     onRender : function(ct, position){
39016         Roo.form.Field.superclass.onRender.call(this, ct, position);
39017         if(!this.el){
39018             var cfg = this.getAutoCreate();
39019             if(!cfg.name){
39020                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39021             }
39022             if (!cfg.name.length) {
39023                 delete cfg.name;
39024             }
39025             if(this.inputType){
39026                 cfg.type = this.inputType;
39027             }
39028             this.el = ct.createChild(cfg, position);
39029         }
39030         var type = this.el.dom.type;
39031         if(type){
39032             if(type == 'password'){
39033                 type = 'text';
39034             }
39035             this.el.addClass('x-form-'+type);
39036         }
39037         if(this.readOnly){
39038             this.el.dom.readOnly = true;
39039         }
39040         if(this.tabIndex !== undefined){
39041             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39042         }
39043
39044         this.el.addClass([this.fieldClass, this.cls]);
39045         this.initValue();
39046     },
39047
39048     /**
39049      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39050      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39051      * @return {Roo.form.Field} this
39052      */
39053     applyTo : function(target){
39054         this.allowDomMove = false;
39055         this.el = Roo.get(target);
39056         this.render(this.el.dom.parentNode);
39057         return this;
39058     },
39059
39060     // private
39061     initValue : function(){
39062         if(this.value !== undefined){
39063             this.setValue(this.value);
39064         }else if(this.el.dom.value.length > 0){
39065             this.setValue(this.el.dom.value);
39066         }
39067     },
39068
39069     /**
39070      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39071      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39072      */
39073     isDirty : function() {
39074         if(this.disabled) {
39075             return false;
39076         }
39077         return String(this.getValue()) !== String(this.originalValue);
39078     },
39079
39080     /**
39081      * stores the current value in loadedValue
39082      */
39083     resetHasChanged : function()
39084     {
39085         this.loadedValue = String(this.getValue());
39086     },
39087     /**
39088      * checks the current value against the 'loaded' value.
39089      * Note - will return false if 'resetHasChanged' has not been called first.
39090      */
39091     hasChanged : function()
39092     {
39093         if(this.disabled || this.readOnly) {
39094             return false;
39095         }
39096         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39097     },
39098     
39099     
39100     
39101     // private
39102     afterRender : function(){
39103         Roo.form.Field.superclass.afterRender.call(this);
39104         this.initEvents();
39105     },
39106
39107     // private
39108     fireKey : function(e){
39109         //Roo.log('field ' + e.getKey());
39110         if(e.isNavKeyPress()){
39111             this.fireEvent("specialkey", this, e);
39112         }
39113     },
39114
39115     /**
39116      * Resets the current field value to the originally loaded value and clears any validation messages
39117      */
39118     reset : function(){
39119         this.setValue(this.resetValue);
39120         this.originalValue = this.getValue();
39121         this.clearInvalid();
39122     },
39123
39124     // private
39125     initEvents : function(){
39126         // safari killled keypress - so keydown is now used..
39127         this.el.on("keydown" , this.fireKey,  this);
39128         this.el.on("focus", this.onFocus,  this);
39129         this.el.on("blur", this.onBlur,  this);
39130         this.el.relayEvent('keyup', this);
39131
39132         // reference to original value for reset
39133         this.originalValue = this.getValue();
39134         this.resetValue =  this.getValue();
39135     },
39136
39137     // private
39138     onFocus : function(){
39139         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39140             this.el.addClass(this.focusClass);
39141         }
39142         if(!this.hasFocus){
39143             this.hasFocus = true;
39144             this.startValue = this.getValue();
39145             this.fireEvent("focus", this);
39146         }
39147     },
39148
39149     beforeBlur : Roo.emptyFn,
39150
39151     // private
39152     onBlur : function(){
39153         this.beforeBlur();
39154         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39155             this.el.removeClass(this.focusClass);
39156         }
39157         this.hasFocus = false;
39158         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39159             this.validate();
39160         }
39161         var v = this.getValue();
39162         if(String(v) !== String(this.startValue)){
39163             this.fireEvent('change', this, v, this.startValue);
39164         }
39165         this.fireEvent("blur", this);
39166     },
39167
39168     /**
39169      * Returns whether or not the field value is currently valid
39170      * @param {Boolean} preventMark True to disable marking the field invalid
39171      * @return {Boolean} True if the value is valid, else false
39172      */
39173     isValid : function(preventMark){
39174         if(this.disabled){
39175             return true;
39176         }
39177         var restore = this.preventMark;
39178         this.preventMark = preventMark === true;
39179         var v = this.validateValue(this.processValue(this.getRawValue()));
39180         this.preventMark = restore;
39181         return v;
39182     },
39183
39184     /**
39185      * Validates the field value
39186      * @return {Boolean} True if the value is valid, else false
39187      */
39188     validate : function(){
39189         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39190             this.clearInvalid();
39191             return true;
39192         }
39193         return false;
39194     },
39195
39196     processValue : function(value){
39197         return value;
39198     },
39199
39200     // private
39201     // Subclasses should provide the validation implementation by overriding this
39202     validateValue : function(value){
39203         return true;
39204     },
39205
39206     /**
39207      * Mark this field as invalid
39208      * @param {String} msg The validation message
39209      */
39210     markInvalid : function(msg){
39211         if(!this.rendered || this.preventMark){ // not rendered
39212             return;
39213         }
39214         
39215         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39216         
39217         obj.el.addClass(this.invalidClass);
39218         msg = msg || this.invalidText;
39219         switch(this.msgTarget){
39220             case 'qtip':
39221                 obj.el.dom.qtip = msg;
39222                 obj.el.dom.qclass = 'x-form-invalid-tip';
39223                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39224                     Roo.QuickTips.enable();
39225                 }
39226                 break;
39227             case 'title':
39228                 this.el.dom.title = msg;
39229                 break;
39230             case 'under':
39231                 if(!this.errorEl){
39232                     var elp = this.el.findParent('.x-form-element', 5, true);
39233                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39234                     this.errorEl.setWidth(elp.getWidth(true)-20);
39235                 }
39236                 this.errorEl.update(msg);
39237                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39238                 break;
39239             case 'side':
39240                 if(!this.errorIcon){
39241                     var elp = this.el.findParent('.x-form-element', 5, true);
39242                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39243                 }
39244                 this.alignErrorIcon();
39245                 this.errorIcon.dom.qtip = msg;
39246                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39247                 this.errorIcon.show();
39248                 this.on('resize', this.alignErrorIcon, this);
39249                 break;
39250             default:
39251                 var t = Roo.getDom(this.msgTarget);
39252                 t.innerHTML = msg;
39253                 t.style.display = this.msgDisplay;
39254                 break;
39255         }
39256         this.fireEvent('invalid', this, msg);
39257     },
39258
39259     // private
39260     alignErrorIcon : function(){
39261         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39262     },
39263
39264     /**
39265      * Clear any invalid styles/messages for this field
39266      */
39267     clearInvalid : function(){
39268         if(!this.rendered || this.preventMark){ // not rendered
39269             return;
39270         }
39271         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39272         
39273         obj.el.removeClass(this.invalidClass);
39274         switch(this.msgTarget){
39275             case 'qtip':
39276                 obj.el.dom.qtip = '';
39277                 break;
39278             case 'title':
39279                 this.el.dom.title = '';
39280                 break;
39281             case 'under':
39282                 if(this.errorEl){
39283                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39284                 }
39285                 break;
39286             case 'side':
39287                 if(this.errorIcon){
39288                     this.errorIcon.dom.qtip = '';
39289                     this.errorIcon.hide();
39290                     this.un('resize', this.alignErrorIcon, this);
39291                 }
39292                 break;
39293             default:
39294                 var t = Roo.getDom(this.msgTarget);
39295                 t.innerHTML = '';
39296                 t.style.display = 'none';
39297                 break;
39298         }
39299         this.fireEvent('valid', this);
39300     },
39301
39302     /**
39303      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39304      * @return {Mixed} value The field value
39305      */
39306     getRawValue : function(){
39307         var v = this.el.getValue();
39308         
39309         return v;
39310     },
39311
39312     /**
39313      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39314      * @return {Mixed} value The field value
39315      */
39316     getValue : function(){
39317         var v = this.el.getValue();
39318          
39319         return v;
39320     },
39321
39322     /**
39323      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39324      * @param {Mixed} value The value to set
39325      */
39326     setRawValue : function(v){
39327         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39328     },
39329
39330     /**
39331      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39332      * @param {Mixed} value The value to set
39333      */
39334     setValue : function(v){
39335         this.value = v;
39336         if(this.rendered){
39337             this.el.dom.value = (v === null || v === undefined ? '' : v);
39338              this.validate();
39339         }
39340     },
39341
39342     adjustSize : function(w, h){
39343         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39344         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39345         return s;
39346     },
39347
39348     adjustWidth : function(tag, w){
39349         tag = tag.toLowerCase();
39350         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39351             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39352                 if(tag == 'input'){
39353                     return w + 2;
39354                 }
39355                 if(tag == 'textarea'){
39356                     return w-2;
39357                 }
39358             }else if(Roo.isOpera){
39359                 if(tag == 'input'){
39360                     return w + 2;
39361                 }
39362                 if(tag == 'textarea'){
39363                     return w-2;
39364                 }
39365             }
39366         }
39367         return w;
39368     }
39369 });
39370
39371
39372 // anything other than normal should be considered experimental
39373 Roo.form.Field.msgFx = {
39374     normal : {
39375         show: function(msgEl, f){
39376             msgEl.setDisplayed('block');
39377         },
39378
39379         hide : function(msgEl, f){
39380             msgEl.setDisplayed(false).update('');
39381         }
39382     },
39383
39384     slide : {
39385         show: function(msgEl, f){
39386             msgEl.slideIn('t', {stopFx:true});
39387         },
39388
39389         hide : function(msgEl, f){
39390             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39391         }
39392     },
39393
39394     slideRight : {
39395         show: function(msgEl, f){
39396             msgEl.fixDisplay();
39397             msgEl.alignTo(f.el, 'tl-tr');
39398             msgEl.slideIn('l', {stopFx:true});
39399         },
39400
39401         hide : function(msgEl, f){
39402             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39403         }
39404     }
39405 };/*
39406  * Based on:
39407  * Ext JS Library 1.1.1
39408  * Copyright(c) 2006-2007, Ext JS, LLC.
39409  *
39410  * Originally Released Under LGPL - original licence link has changed is not relivant.
39411  *
39412  * Fork - LGPL
39413  * <script type="text/javascript">
39414  */
39415  
39416
39417 /**
39418  * @class Roo.form.TextField
39419  * @extends Roo.form.Field
39420  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39421  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39422  * @constructor
39423  * Creates a new TextField
39424  * @param {Object} config Configuration options
39425  */
39426 Roo.form.TextField = function(config){
39427     Roo.form.TextField.superclass.constructor.call(this, config);
39428     this.addEvents({
39429         /**
39430          * @event autosize
39431          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39432          * according to the default logic, but this event provides a hook for the developer to apply additional
39433          * logic at runtime to resize the field if needed.
39434              * @param {Roo.form.Field} this This text field
39435              * @param {Number} width The new field width
39436              */
39437         autosize : true
39438     });
39439 };
39440
39441 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39442     /**
39443      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39444      */
39445     grow : false,
39446     /**
39447      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39448      */
39449     growMin : 30,
39450     /**
39451      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39452      */
39453     growMax : 800,
39454     /**
39455      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39456      */
39457     vtype : null,
39458     /**
39459      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39460      */
39461     maskRe : null,
39462     /**
39463      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39464      */
39465     disableKeyFilter : false,
39466     /**
39467      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39468      */
39469     allowBlank : true,
39470     /**
39471      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39472      */
39473     minLength : 0,
39474     /**
39475      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39476      */
39477     maxLength : Number.MAX_VALUE,
39478     /**
39479      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39480      */
39481     minLengthText : "The minimum length for this field is {0}",
39482     /**
39483      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39484      */
39485     maxLengthText : "The maximum length for this field is {0}",
39486     /**
39487      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39488      */
39489     selectOnFocus : false,
39490     /**
39491      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39492      */    
39493     allowLeadingSpace : false,
39494     /**
39495      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39496      */
39497     blankText : "This field is required",
39498     /**
39499      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39500      * If available, this function will be called only after the basic validators all return true, and will be passed the
39501      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39502      */
39503     validator : null,
39504     /**
39505      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39506      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39507      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39508      */
39509     regex : null,
39510     /**
39511      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39512      */
39513     regexText : "",
39514     /**
39515      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39516      */
39517     emptyText : null,
39518    
39519
39520     // private
39521     initEvents : function()
39522     {
39523         if (this.emptyText) {
39524             this.el.attr('placeholder', this.emptyText);
39525         }
39526         
39527         Roo.form.TextField.superclass.initEvents.call(this);
39528         if(this.validationEvent == 'keyup'){
39529             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39530             this.el.on('keyup', this.filterValidation, this);
39531         }
39532         else if(this.validationEvent !== false){
39533             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39534         }
39535         
39536         if(this.selectOnFocus){
39537             this.on("focus", this.preFocus, this);
39538         }
39539         if (!this.allowLeadingSpace) {
39540             this.on('blur', this.cleanLeadingSpace, this);
39541         }
39542         
39543         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39544             this.el.on("keypress", this.filterKeys, this);
39545         }
39546         if(this.grow){
39547             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39548             this.el.on("click", this.autoSize,  this);
39549         }
39550         if(this.el.is('input[type=password]') && Roo.isSafari){
39551             this.el.on('keydown', this.SafariOnKeyDown, this);
39552         }
39553     },
39554
39555     processValue : function(value){
39556         if(this.stripCharsRe){
39557             var newValue = value.replace(this.stripCharsRe, '');
39558             if(newValue !== value){
39559                 this.setRawValue(newValue);
39560                 return newValue;
39561             }
39562         }
39563         return value;
39564     },
39565
39566     filterValidation : function(e){
39567         if(!e.isNavKeyPress()){
39568             this.validationTask.delay(this.validationDelay);
39569         }
39570     },
39571
39572     // private
39573     onKeyUp : function(e){
39574         if(!e.isNavKeyPress()){
39575             this.autoSize();
39576         }
39577     },
39578     // private - clean the leading white space
39579     cleanLeadingSpace : function(e)
39580     {
39581         if ( this.inputType == 'file') {
39582             return;
39583         }
39584         
39585         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39586     },
39587     /**
39588      * Resets the current field value to the originally-loaded value and clears any validation messages.
39589      *  
39590      */
39591     reset : function(){
39592         Roo.form.TextField.superclass.reset.call(this);
39593        
39594     }, 
39595     // private
39596     preFocus : function(){
39597         
39598         if(this.selectOnFocus){
39599             this.el.dom.select();
39600         }
39601     },
39602
39603     
39604     // private
39605     filterKeys : function(e){
39606         var k = e.getKey();
39607         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39608             return;
39609         }
39610         var c = e.getCharCode(), cc = String.fromCharCode(c);
39611         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39612             return;
39613         }
39614         if(!this.maskRe.test(cc)){
39615             e.stopEvent();
39616         }
39617     },
39618
39619     setValue : function(v){
39620         
39621         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39622         
39623         this.autoSize();
39624     },
39625
39626     /**
39627      * Validates a value according to the field's validation rules and marks the field as invalid
39628      * if the validation fails
39629      * @param {Mixed} value The value to validate
39630      * @return {Boolean} True if the value is valid, else false
39631      */
39632     validateValue : function(value){
39633         if(value.length < 1)  { // if it's blank
39634              if(this.allowBlank){
39635                 this.clearInvalid();
39636                 return true;
39637              }else{
39638                 this.markInvalid(this.blankText);
39639                 return false;
39640              }
39641         }
39642         if(value.length < this.minLength){
39643             this.markInvalid(String.format(this.minLengthText, this.minLength));
39644             return false;
39645         }
39646         if(value.length > this.maxLength){
39647             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39648             return false;
39649         }
39650         if(this.vtype){
39651             var vt = Roo.form.VTypes;
39652             if(!vt[this.vtype](value, this)){
39653                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39654                 return false;
39655             }
39656         }
39657         if(typeof this.validator == "function"){
39658             var msg = this.validator(value);
39659             if(msg !== true){
39660                 this.markInvalid(msg);
39661                 return false;
39662             }
39663         }
39664         if(this.regex && !this.regex.test(value)){
39665             this.markInvalid(this.regexText);
39666             return false;
39667         }
39668         return true;
39669     },
39670
39671     /**
39672      * Selects text in this field
39673      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39674      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39675      */
39676     selectText : function(start, end){
39677         var v = this.getRawValue();
39678         if(v.length > 0){
39679             start = start === undefined ? 0 : start;
39680             end = end === undefined ? v.length : end;
39681             var d = this.el.dom;
39682             if(d.setSelectionRange){
39683                 d.setSelectionRange(start, end);
39684             }else if(d.createTextRange){
39685                 var range = d.createTextRange();
39686                 range.moveStart("character", start);
39687                 range.moveEnd("character", v.length-end);
39688                 range.select();
39689             }
39690         }
39691     },
39692
39693     /**
39694      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39695      * This only takes effect if grow = true, and fires the autosize event.
39696      */
39697     autoSize : function(){
39698         if(!this.grow || !this.rendered){
39699             return;
39700         }
39701         if(!this.metrics){
39702             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39703         }
39704         var el = this.el;
39705         var v = el.dom.value;
39706         var d = document.createElement('div');
39707         d.appendChild(document.createTextNode(v));
39708         v = d.innerHTML;
39709         d = null;
39710         v += "&#160;";
39711         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39712         this.el.setWidth(w);
39713         this.fireEvent("autosize", this, w);
39714     },
39715     
39716     // private
39717     SafariOnKeyDown : function(event)
39718     {
39719         // this is a workaround for a password hang bug on chrome/ webkit.
39720         
39721         var isSelectAll = false;
39722         
39723         if(this.el.dom.selectionEnd > 0){
39724             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39725         }
39726         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39727             event.preventDefault();
39728             this.setValue('');
39729             return;
39730         }
39731         
39732         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39733             
39734             event.preventDefault();
39735             // this is very hacky as keydown always get's upper case.
39736             
39737             var cc = String.fromCharCode(event.getCharCode());
39738             
39739             
39740             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39741             
39742         }
39743         
39744         
39745     }
39746 });/*
39747  * Based on:
39748  * Ext JS Library 1.1.1
39749  * Copyright(c) 2006-2007, Ext JS, LLC.
39750  *
39751  * Originally Released Under LGPL - original licence link has changed is not relivant.
39752  *
39753  * Fork - LGPL
39754  * <script type="text/javascript">
39755  */
39756  
39757 /**
39758  * @class Roo.form.Hidden
39759  * @extends Roo.form.TextField
39760  * Simple Hidden element used on forms 
39761  * 
39762  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39763  * 
39764  * @constructor
39765  * Creates a new Hidden form element.
39766  * @param {Object} config Configuration options
39767  */
39768
39769
39770
39771 // easy hidden field...
39772 Roo.form.Hidden = function(config){
39773     Roo.form.Hidden.superclass.constructor.call(this, config);
39774 };
39775   
39776 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39777     fieldLabel:      '',
39778     inputType:      'hidden',
39779     width:          50,
39780     allowBlank:     true,
39781     labelSeparator: '',
39782     hidden:         true,
39783     itemCls :       'x-form-item-display-none'
39784
39785
39786 });
39787
39788
39789 /*
39790  * Based on:
39791  * Ext JS Library 1.1.1
39792  * Copyright(c) 2006-2007, Ext JS, LLC.
39793  *
39794  * Originally Released Under LGPL - original licence link has changed is not relivant.
39795  *
39796  * Fork - LGPL
39797  * <script type="text/javascript">
39798  */
39799  
39800 /**
39801  * @class Roo.form.TriggerField
39802  * @extends Roo.form.TextField
39803  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39804  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39805  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39806  * for which you can provide a custom implementation.  For example:
39807  * <pre><code>
39808 var trigger = new Roo.form.TriggerField();
39809 trigger.onTriggerClick = myTriggerFn;
39810 trigger.applyTo('my-field');
39811 </code></pre>
39812  *
39813  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39814  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39815  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39816  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39817  * @constructor
39818  * Create a new TriggerField.
39819  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39820  * to the base TextField)
39821  */
39822 Roo.form.TriggerField = function(config){
39823     this.mimicing = false;
39824     Roo.form.TriggerField.superclass.constructor.call(this, config);
39825 };
39826
39827 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39828     /**
39829      * @cfg {String} triggerClass A CSS class to apply to the trigger
39830      */
39831     /**
39832      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39833      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39834      */
39835     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39836     /**
39837      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39838      */
39839     hideTrigger:false,
39840
39841     /** @cfg {Boolean} grow @hide */
39842     /** @cfg {Number} growMin @hide */
39843     /** @cfg {Number} growMax @hide */
39844
39845     /**
39846      * @hide 
39847      * @method
39848      */
39849     autoSize: Roo.emptyFn,
39850     // private
39851     monitorTab : true,
39852     // private
39853     deferHeight : true,
39854
39855     
39856     actionMode : 'wrap',
39857     // private
39858     onResize : function(w, h){
39859         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39860         if(typeof w == 'number'){
39861             var x = w - this.trigger.getWidth();
39862             this.el.setWidth(this.adjustWidth('input', x));
39863             this.trigger.setStyle('left', x+'px');
39864         }
39865     },
39866
39867     // private
39868     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39869
39870     // private
39871     getResizeEl : function(){
39872         return this.wrap;
39873     },
39874
39875     // private
39876     getPositionEl : function(){
39877         return this.wrap;
39878     },
39879
39880     // private
39881     alignErrorIcon : function(){
39882         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39883     },
39884
39885     // private
39886     onRender : function(ct, position){
39887         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39888         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39889         this.trigger = this.wrap.createChild(this.triggerConfig ||
39890                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39891         if(this.hideTrigger){
39892             this.trigger.setDisplayed(false);
39893         }
39894         this.initTrigger();
39895         if(!this.width){
39896             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39897         }
39898     },
39899
39900     // private
39901     initTrigger : function(){
39902         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39903         this.trigger.addClassOnOver('x-form-trigger-over');
39904         this.trigger.addClassOnClick('x-form-trigger-click');
39905     },
39906
39907     // private
39908     onDestroy : function(){
39909         if(this.trigger){
39910             this.trigger.removeAllListeners();
39911             this.trigger.remove();
39912         }
39913         if(this.wrap){
39914             this.wrap.remove();
39915         }
39916         Roo.form.TriggerField.superclass.onDestroy.call(this);
39917     },
39918
39919     // private
39920     onFocus : function(){
39921         Roo.form.TriggerField.superclass.onFocus.call(this);
39922         if(!this.mimicing){
39923             this.wrap.addClass('x-trigger-wrap-focus');
39924             this.mimicing = true;
39925             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39926             if(this.monitorTab){
39927                 this.el.on("keydown", this.checkTab, this);
39928             }
39929         }
39930     },
39931
39932     // private
39933     checkTab : function(e){
39934         if(e.getKey() == e.TAB){
39935             this.triggerBlur();
39936         }
39937     },
39938
39939     // private
39940     onBlur : function(){
39941         // do nothing
39942     },
39943
39944     // private
39945     mimicBlur : function(e, t){
39946         if(!this.wrap.contains(t) && this.validateBlur()){
39947             this.triggerBlur();
39948         }
39949     },
39950
39951     // private
39952     triggerBlur : function(){
39953         this.mimicing = false;
39954         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39955         if(this.monitorTab){
39956             this.el.un("keydown", this.checkTab, this);
39957         }
39958         this.wrap.removeClass('x-trigger-wrap-focus');
39959         Roo.form.TriggerField.superclass.onBlur.call(this);
39960     },
39961
39962     // private
39963     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39964     validateBlur : function(e, t){
39965         return true;
39966     },
39967
39968     // private
39969     onDisable : function(){
39970         Roo.form.TriggerField.superclass.onDisable.call(this);
39971         if(this.wrap){
39972             this.wrap.addClass('x-item-disabled');
39973         }
39974     },
39975
39976     // private
39977     onEnable : function(){
39978         Roo.form.TriggerField.superclass.onEnable.call(this);
39979         if(this.wrap){
39980             this.wrap.removeClass('x-item-disabled');
39981         }
39982     },
39983
39984     // private
39985     onShow : function(){
39986         var ae = this.getActionEl();
39987         
39988         if(ae){
39989             ae.dom.style.display = '';
39990             ae.dom.style.visibility = 'visible';
39991         }
39992     },
39993
39994     // private
39995     
39996     onHide : function(){
39997         var ae = this.getActionEl();
39998         ae.dom.style.display = 'none';
39999     },
40000
40001     /**
40002      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
40003      * by an implementing function.
40004      * @method
40005      * @param {EventObject} e
40006      */
40007     onTriggerClick : Roo.emptyFn
40008 });
40009
40010 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40011 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40012 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40013 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40014     initComponent : function(){
40015         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40016
40017         this.triggerConfig = {
40018             tag:'span', cls:'x-form-twin-triggers', cn:[
40019             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40020             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40021         ]};
40022     },
40023
40024     getTrigger : function(index){
40025         return this.triggers[index];
40026     },
40027
40028     initTrigger : function(){
40029         var ts = this.trigger.select('.x-form-trigger', true);
40030         this.wrap.setStyle('overflow', 'hidden');
40031         var triggerField = this;
40032         ts.each(function(t, all, index){
40033             t.hide = function(){
40034                 var w = triggerField.wrap.getWidth();
40035                 this.dom.style.display = 'none';
40036                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40037             };
40038             t.show = function(){
40039                 var w = triggerField.wrap.getWidth();
40040                 this.dom.style.display = '';
40041                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40042             };
40043             var triggerIndex = 'Trigger'+(index+1);
40044
40045             if(this['hide'+triggerIndex]){
40046                 t.dom.style.display = 'none';
40047             }
40048             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40049             t.addClassOnOver('x-form-trigger-over');
40050             t.addClassOnClick('x-form-trigger-click');
40051         }, this);
40052         this.triggers = ts.elements;
40053     },
40054
40055     onTrigger1Click : Roo.emptyFn,
40056     onTrigger2Click : Roo.emptyFn
40057 });/*
40058  * Based on:
40059  * Ext JS Library 1.1.1
40060  * Copyright(c) 2006-2007, Ext JS, LLC.
40061  *
40062  * Originally Released Under LGPL - original licence link has changed is not relivant.
40063  *
40064  * Fork - LGPL
40065  * <script type="text/javascript">
40066  */
40067  
40068 /**
40069  * @class Roo.form.TextArea
40070  * @extends Roo.form.TextField
40071  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40072  * support for auto-sizing.
40073  * @constructor
40074  * Creates a new TextArea
40075  * @param {Object} config Configuration options
40076  */
40077 Roo.form.TextArea = function(config){
40078     Roo.form.TextArea.superclass.constructor.call(this, config);
40079     // these are provided exchanges for backwards compat
40080     // minHeight/maxHeight were replaced by growMin/growMax to be
40081     // compatible with TextField growing config values
40082     if(this.minHeight !== undefined){
40083         this.growMin = this.minHeight;
40084     }
40085     if(this.maxHeight !== undefined){
40086         this.growMax = this.maxHeight;
40087     }
40088 };
40089
40090 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40091     /**
40092      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40093      */
40094     growMin : 60,
40095     /**
40096      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40097      */
40098     growMax: 1000,
40099     /**
40100      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40101      * in the field (equivalent to setting overflow: hidden, defaults to false)
40102      */
40103     preventScrollbars: false,
40104     /**
40105      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40106      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40107      */
40108
40109     // private
40110     onRender : function(ct, position){
40111         if(!this.el){
40112             this.defaultAutoCreate = {
40113                 tag: "textarea",
40114                 style:"width:300px;height:60px;",
40115                 autocomplete: "new-password"
40116             };
40117         }
40118         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40119         if(this.grow){
40120             this.textSizeEl = Roo.DomHelper.append(document.body, {
40121                 tag: "pre", cls: "x-form-grow-sizer"
40122             });
40123             if(this.preventScrollbars){
40124                 this.el.setStyle("overflow", "hidden");
40125             }
40126             this.el.setHeight(this.growMin);
40127         }
40128     },
40129
40130     onDestroy : function(){
40131         if(this.textSizeEl){
40132             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40133         }
40134         Roo.form.TextArea.superclass.onDestroy.call(this);
40135     },
40136
40137     // private
40138     onKeyUp : function(e){
40139         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40140             this.autoSize();
40141         }
40142     },
40143
40144     /**
40145      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40146      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40147      */
40148     autoSize : function(){
40149         if(!this.grow || !this.textSizeEl){
40150             return;
40151         }
40152         var el = this.el;
40153         var v = el.dom.value;
40154         var ts = this.textSizeEl;
40155
40156         ts.innerHTML = '';
40157         ts.appendChild(document.createTextNode(v));
40158         v = ts.innerHTML;
40159
40160         Roo.fly(ts).setWidth(this.el.getWidth());
40161         if(v.length < 1){
40162             v = "&#160;&#160;";
40163         }else{
40164             if(Roo.isIE){
40165                 v = v.replace(/\n/g, '<p>&#160;</p>');
40166             }
40167             v += "&#160;\n&#160;";
40168         }
40169         ts.innerHTML = v;
40170         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40171         if(h != this.lastHeight){
40172             this.lastHeight = h;
40173             this.el.setHeight(h);
40174             this.fireEvent("autosize", this, h);
40175         }
40176     }
40177 });/*
40178  * Based on:
40179  * Ext JS Library 1.1.1
40180  * Copyright(c) 2006-2007, Ext JS, LLC.
40181  *
40182  * Originally Released Under LGPL - original licence link has changed is not relivant.
40183  *
40184  * Fork - LGPL
40185  * <script type="text/javascript">
40186  */
40187  
40188
40189 /**
40190  * @class Roo.form.NumberField
40191  * @extends Roo.form.TextField
40192  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40193  * @constructor
40194  * Creates a new NumberField
40195  * @param {Object} config Configuration options
40196  */
40197 Roo.form.NumberField = function(config){
40198     Roo.form.NumberField.superclass.constructor.call(this, config);
40199 };
40200
40201 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40202     /**
40203      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40204      */
40205     fieldClass: "x-form-field x-form-num-field",
40206     /**
40207      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40208      */
40209     allowDecimals : true,
40210     /**
40211      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40212      */
40213     decimalSeparator : ".",
40214     /**
40215      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40216      */
40217     decimalPrecision : 2,
40218     /**
40219      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40220      */
40221     allowNegative : true,
40222     /**
40223      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40224      */
40225     minValue : Number.NEGATIVE_INFINITY,
40226     /**
40227      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40228      */
40229     maxValue : Number.MAX_VALUE,
40230     /**
40231      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40232      */
40233     minText : "The minimum value for this field is {0}",
40234     /**
40235      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40236      */
40237     maxText : "The maximum value for this field is {0}",
40238     /**
40239      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40240      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40241      */
40242     nanText : "{0} is not a valid number",
40243
40244     // private
40245     initEvents : function(){
40246         Roo.form.NumberField.superclass.initEvents.call(this);
40247         var allowed = "0123456789";
40248         if(this.allowDecimals){
40249             allowed += this.decimalSeparator;
40250         }
40251         if(this.allowNegative){
40252             allowed += "-";
40253         }
40254         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40255         var keyPress = function(e){
40256             var k = e.getKey();
40257             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40258                 return;
40259             }
40260             var c = e.getCharCode();
40261             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40262                 e.stopEvent();
40263             }
40264         };
40265         this.el.on("keypress", keyPress, this);
40266     },
40267
40268     // private
40269     validateValue : function(value){
40270         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40271             return false;
40272         }
40273         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40274              return true;
40275         }
40276         var num = this.parseValue(value);
40277         if(isNaN(num)){
40278             this.markInvalid(String.format(this.nanText, value));
40279             return false;
40280         }
40281         if(num < this.minValue){
40282             this.markInvalid(String.format(this.minText, this.minValue));
40283             return false;
40284         }
40285         if(num > this.maxValue){
40286             this.markInvalid(String.format(this.maxText, this.maxValue));
40287             return false;
40288         }
40289         return true;
40290     },
40291
40292     getValue : function(){
40293         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40294     },
40295
40296     // private
40297     parseValue : function(value){
40298         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40299         return isNaN(value) ? '' : value;
40300     },
40301
40302     // private
40303     fixPrecision : function(value){
40304         var nan = isNaN(value);
40305         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40306             return nan ? '' : value;
40307         }
40308         return parseFloat(value).toFixed(this.decimalPrecision);
40309     },
40310
40311     setValue : function(v){
40312         v = this.fixPrecision(v);
40313         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40314     },
40315
40316     // private
40317     decimalPrecisionFcn : function(v){
40318         return Math.floor(v);
40319     },
40320
40321     beforeBlur : function(){
40322         var v = this.parseValue(this.getRawValue());
40323         if(v){
40324             this.setValue(v);
40325         }
40326     }
40327 });/*
40328  * Based on:
40329  * Ext JS Library 1.1.1
40330  * Copyright(c) 2006-2007, Ext JS, LLC.
40331  *
40332  * Originally Released Under LGPL - original licence link has changed is not relivant.
40333  *
40334  * Fork - LGPL
40335  * <script type="text/javascript">
40336  */
40337  
40338 /**
40339  * @class Roo.form.DateField
40340  * @extends Roo.form.TriggerField
40341  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40342 * @constructor
40343 * Create a new DateField
40344 * @param {Object} config
40345  */
40346 Roo.form.DateField = function(config)
40347 {
40348     Roo.form.DateField.superclass.constructor.call(this, config);
40349     
40350       this.addEvents({
40351          
40352         /**
40353          * @event select
40354          * Fires when a date is selected
40355              * @param {Roo.form.DateField} combo This combo box
40356              * @param {Date} date The date selected
40357              */
40358         'select' : true
40359          
40360     });
40361     
40362     
40363     if(typeof this.minValue == "string") {
40364         this.minValue = this.parseDate(this.minValue);
40365     }
40366     if(typeof this.maxValue == "string") {
40367         this.maxValue = this.parseDate(this.maxValue);
40368     }
40369     this.ddMatch = null;
40370     if(this.disabledDates){
40371         var dd = this.disabledDates;
40372         var re = "(?:";
40373         for(var i = 0; i < dd.length; i++){
40374             re += dd[i];
40375             if(i != dd.length-1) {
40376                 re += "|";
40377             }
40378         }
40379         this.ddMatch = new RegExp(re + ")");
40380     }
40381 };
40382
40383 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40384     /**
40385      * @cfg {String} format
40386      * The default date format string which can be overriden for localization support.  The format must be
40387      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40388      */
40389     format : "m/d/y",
40390     /**
40391      * @cfg {String} altFormats
40392      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40393      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40394      */
40395     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40396     /**
40397      * @cfg {Array} disabledDays
40398      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40399      */
40400     disabledDays : null,
40401     /**
40402      * @cfg {String} disabledDaysText
40403      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40404      */
40405     disabledDaysText : "Disabled",
40406     /**
40407      * @cfg {Array} disabledDates
40408      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40409      * expression so they are very powerful. Some examples:
40410      * <ul>
40411      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40412      * <li>["03/08", "09/16"] would disable those days for every year</li>
40413      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40414      * <li>["03/../2006"] would disable every day in March 2006</li>
40415      * <li>["^03"] would disable every day in every March</li>
40416      * </ul>
40417      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40418      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40419      */
40420     disabledDates : null,
40421     /**
40422      * @cfg {String} disabledDatesText
40423      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40424      */
40425     disabledDatesText : "Disabled",
40426     /**
40427      * @cfg {Date/String} minValue
40428      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40429      * valid format (defaults to null).
40430      */
40431     minValue : null,
40432     /**
40433      * @cfg {Date/String} maxValue
40434      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40435      * valid format (defaults to null).
40436      */
40437     maxValue : null,
40438     /**
40439      * @cfg {String} minText
40440      * The error text to display when the date in the cell is before minValue (defaults to
40441      * 'The date in this field must be after {minValue}').
40442      */
40443     minText : "The date in this field must be equal to or after {0}",
40444     /**
40445      * @cfg {String} maxText
40446      * The error text to display when the date in the cell is after maxValue (defaults to
40447      * 'The date in this field must be before {maxValue}').
40448      */
40449     maxText : "The date in this field must be equal to or before {0}",
40450     /**
40451      * @cfg {String} invalidText
40452      * The error text to display when the date in the field is invalid (defaults to
40453      * '{value} is not a valid date - it must be in the format {format}').
40454      */
40455     invalidText : "{0} is not a valid date - it must be in the format {1}",
40456     /**
40457      * @cfg {String} triggerClass
40458      * An additional CSS class used to style the trigger button.  The trigger will always get the
40459      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40460      * which displays a calendar icon).
40461      */
40462     triggerClass : 'x-form-date-trigger',
40463     
40464
40465     /**
40466      * @cfg {Boolean} useIso
40467      * if enabled, then the date field will use a hidden field to store the 
40468      * real value as iso formated date. default (false)
40469      */ 
40470     useIso : false,
40471     /**
40472      * @cfg {String/Object} autoCreate
40473      * A DomHelper element spec, or true for a default element spec (defaults to
40474      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40475      */ 
40476     // private
40477     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40478     
40479     // private
40480     hiddenField: false,
40481     
40482     onRender : function(ct, position)
40483     {
40484         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40485         if (this.useIso) {
40486             //this.el.dom.removeAttribute('name'); 
40487             Roo.log("Changing name?");
40488             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40489             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40490                     'before', true);
40491             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40492             // prevent input submission
40493             this.hiddenName = this.name;
40494         }
40495             
40496             
40497     },
40498     
40499     // private
40500     validateValue : function(value)
40501     {
40502         value = this.formatDate(value);
40503         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40504             Roo.log('super failed');
40505             return false;
40506         }
40507         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40508              return true;
40509         }
40510         var svalue = value;
40511         value = this.parseDate(value);
40512         if(!value){
40513             Roo.log('parse date failed' + svalue);
40514             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40515             return false;
40516         }
40517         var time = value.getTime();
40518         if(this.minValue && time < this.minValue.getTime()){
40519             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40520             return false;
40521         }
40522         if(this.maxValue && time > this.maxValue.getTime()){
40523             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40524             return false;
40525         }
40526         if(this.disabledDays){
40527             var day = value.getDay();
40528             for(var i = 0; i < this.disabledDays.length; i++) {
40529                 if(day === this.disabledDays[i]){
40530                     this.markInvalid(this.disabledDaysText);
40531                     return false;
40532                 }
40533             }
40534         }
40535         var fvalue = this.formatDate(value);
40536         if(this.ddMatch && this.ddMatch.test(fvalue)){
40537             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40538             return false;
40539         }
40540         return true;
40541     },
40542
40543     // private
40544     // Provides logic to override the default TriggerField.validateBlur which just returns true
40545     validateBlur : function(){
40546         return !this.menu || !this.menu.isVisible();
40547     },
40548     
40549     getName: function()
40550     {
40551         // returns hidden if it's set..
40552         if (!this.rendered) {return ''};
40553         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40554         
40555     },
40556
40557     /**
40558      * Returns the current date value of the date field.
40559      * @return {Date} The date value
40560      */
40561     getValue : function(){
40562         
40563         return  this.hiddenField ?
40564                 this.hiddenField.value :
40565                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40566     },
40567
40568     /**
40569      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40570      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40571      * (the default format used is "m/d/y").
40572      * <br />Usage:
40573      * <pre><code>
40574 //All of these calls set the same date value (May 4, 2006)
40575
40576 //Pass a date object:
40577 var dt = new Date('5/4/06');
40578 dateField.setValue(dt);
40579
40580 //Pass a date string (default format):
40581 dateField.setValue('5/4/06');
40582
40583 //Pass a date string (custom format):
40584 dateField.format = 'Y-m-d';
40585 dateField.setValue('2006-5-4');
40586 </code></pre>
40587      * @param {String/Date} date The date or valid date string
40588      */
40589     setValue : function(date){
40590         if (this.hiddenField) {
40591             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40592         }
40593         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40594         // make sure the value field is always stored as a date..
40595         this.value = this.parseDate(date);
40596         
40597         
40598     },
40599
40600     // private
40601     parseDate : function(value){
40602         if(!value || value instanceof Date){
40603             return value;
40604         }
40605         var v = Date.parseDate(value, this.format);
40606          if (!v && this.useIso) {
40607             v = Date.parseDate(value, 'Y-m-d');
40608         }
40609         if(!v && this.altFormats){
40610             if(!this.altFormatsArray){
40611                 this.altFormatsArray = this.altFormats.split("|");
40612             }
40613             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40614                 v = Date.parseDate(value, this.altFormatsArray[i]);
40615             }
40616         }
40617         return v;
40618     },
40619
40620     // private
40621     formatDate : function(date, fmt){
40622         return (!date || !(date instanceof Date)) ?
40623                date : date.dateFormat(fmt || this.format);
40624     },
40625
40626     // private
40627     menuListeners : {
40628         select: function(m, d){
40629             
40630             this.setValue(d);
40631             this.fireEvent('select', this, d);
40632         },
40633         show : function(){ // retain focus styling
40634             this.onFocus();
40635         },
40636         hide : function(){
40637             this.focus.defer(10, this);
40638             var ml = this.menuListeners;
40639             this.menu.un("select", ml.select,  this);
40640             this.menu.un("show", ml.show,  this);
40641             this.menu.un("hide", ml.hide,  this);
40642         }
40643     },
40644
40645     // private
40646     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40647     onTriggerClick : function(){
40648         if(this.disabled){
40649             return;
40650         }
40651         if(this.menu == null){
40652             this.menu = new Roo.menu.DateMenu();
40653         }
40654         Roo.apply(this.menu.picker,  {
40655             showClear: this.allowBlank,
40656             minDate : this.minValue,
40657             maxDate : this.maxValue,
40658             disabledDatesRE : this.ddMatch,
40659             disabledDatesText : this.disabledDatesText,
40660             disabledDays : this.disabledDays,
40661             disabledDaysText : this.disabledDaysText,
40662             format : this.useIso ? 'Y-m-d' : this.format,
40663             minText : String.format(this.minText, this.formatDate(this.minValue)),
40664             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40665         });
40666         this.menu.on(Roo.apply({}, this.menuListeners, {
40667             scope:this
40668         }));
40669         this.menu.picker.setValue(this.getValue() || new Date());
40670         this.menu.show(this.el, "tl-bl?");
40671     },
40672
40673     beforeBlur : function(){
40674         var v = this.parseDate(this.getRawValue());
40675         if(v){
40676             this.setValue(v);
40677         }
40678     },
40679
40680     /*@
40681      * overide
40682      * 
40683      */
40684     isDirty : function() {
40685         if(this.disabled) {
40686             return false;
40687         }
40688         
40689         if(typeof(this.startValue) === 'undefined'){
40690             return false;
40691         }
40692         
40693         return String(this.getValue()) !== String(this.startValue);
40694         
40695     },
40696     // @overide
40697     cleanLeadingSpace : function(e)
40698     {
40699        return;
40700     }
40701     
40702 });/*
40703  * Based on:
40704  * Ext JS Library 1.1.1
40705  * Copyright(c) 2006-2007, Ext JS, LLC.
40706  *
40707  * Originally Released Under LGPL - original licence link has changed is not relivant.
40708  *
40709  * Fork - LGPL
40710  * <script type="text/javascript">
40711  */
40712  
40713 /**
40714  * @class Roo.form.MonthField
40715  * @extends Roo.form.TriggerField
40716  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40717 * @constructor
40718 * Create a new MonthField
40719 * @param {Object} config
40720  */
40721 Roo.form.MonthField = function(config){
40722     
40723     Roo.form.MonthField.superclass.constructor.call(this, config);
40724     
40725       this.addEvents({
40726          
40727         /**
40728          * @event select
40729          * Fires when a date is selected
40730              * @param {Roo.form.MonthFieeld} combo This combo box
40731              * @param {Date} date The date selected
40732              */
40733         'select' : true
40734          
40735     });
40736     
40737     
40738     if(typeof this.minValue == "string") {
40739         this.minValue = this.parseDate(this.minValue);
40740     }
40741     if(typeof this.maxValue == "string") {
40742         this.maxValue = this.parseDate(this.maxValue);
40743     }
40744     this.ddMatch = null;
40745     if(this.disabledDates){
40746         var dd = this.disabledDates;
40747         var re = "(?:";
40748         for(var i = 0; i < dd.length; i++){
40749             re += dd[i];
40750             if(i != dd.length-1) {
40751                 re += "|";
40752             }
40753         }
40754         this.ddMatch = new RegExp(re + ")");
40755     }
40756 };
40757
40758 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40759     /**
40760      * @cfg {String} format
40761      * The default date format string which can be overriden for localization support.  The format must be
40762      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40763      */
40764     format : "M Y",
40765     /**
40766      * @cfg {String} altFormats
40767      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40768      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40769      */
40770     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40771     /**
40772      * @cfg {Array} disabledDays
40773      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40774      */
40775     disabledDays : [0,1,2,3,4,5,6],
40776     /**
40777      * @cfg {String} disabledDaysText
40778      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40779      */
40780     disabledDaysText : "Disabled",
40781     /**
40782      * @cfg {Array} disabledDates
40783      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40784      * expression so they are very powerful. Some examples:
40785      * <ul>
40786      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40787      * <li>["03/08", "09/16"] would disable those days for every year</li>
40788      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40789      * <li>["03/../2006"] would disable every day in March 2006</li>
40790      * <li>["^03"] would disable every day in every March</li>
40791      * </ul>
40792      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40793      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40794      */
40795     disabledDates : null,
40796     /**
40797      * @cfg {String} disabledDatesText
40798      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40799      */
40800     disabledDatesText : "Disabled",
40801     /**
40802      * @cfg {Date/String} minValue
40803      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40804      * valid format (defaults to null).
40805      */
40806     minValue : null,
40807     /**
40808      * @cfg {Date/String} maxValue
40809      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40810      * valid format (defaults to null).
40811      */
40812     maxValue : null,
40813     /**
40814      * @cfg {String} minText
40815      * The error text to display when the date in the cell is before minValue (defaults to
40816      * 'The date in this field must be after {minValue}').
40817      */
40818     minText : "The date in this field must be equal to or after {0}",
40819     /**
40820      * @cfg {String} maxTextf
40821      * The error text to display when the date in the cell is after maxValue (defaults to
40822      * 'The date in this field must be before {maxValue}').
40823      */
40824     maxText : "The date in this field must be equal to or before {0}",
40825     /**
40826      * @cfg {String} invalidText
40827      * The error text to display when the date in the field is invalid (defaults to
40828      * '{value} is not a valid date - it must be in the format {format}').
40829      */
40830     invalidText : "{0} is not a valid date - it must be in the format {1}",
40831     /**
40832      * @cfg {String} triggerClass
40833      * An additional CSS class used to style the trigger button.  The trigger will always get the
40834      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40835      * which displays a calendar icon).
40836      */
40837     triggerClass : 'x-form-date-trigger',
40838     
40839
40840     /**
40841      * @cfg {Boolean} useIso
40842      * if enabled, then the date field will use a hidden field to store the 
40843      * real value as iso formated date. default (true)
40844      */ 
40845     useIso : true,
40846     /**
40847      * @cfg {String/Object} autoCreate
40848      * A DomHelper element spec, or true for a default element spec (defaults to
40849      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40850      */ 
40851     // private
40852     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40853     
40854     // private
40855     hiddenField: false,
40856     
40857     hideMonthPicker : false,
40858     
40859     onRender : function(ct, position)
40860     {
40861         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40862         if (this.useIso) {
40863             this.el.dom.removeAttribute('name'); 
40864             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40865                     'before', true);
40866             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40867             // prevent input submission
40868             this.hiddenName = this.name;
40869         }
40870             
40871             
40872     },
40873     
40874     // private
40875     validateValue : function(value)
40876     {
40877         value = this.formatDate(value);
40878         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40879             return false;
40880         }
40881         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40882              return true;
40883         }
40884         var svalue = value;
40885         value = this.parseDate(value);
40886         if(!value){
40887             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40888             return false;
40889         }
40890         var time = value.getTime();
40891         if(this.minValue && time < this.minValue.getTime()){
40892             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40893             return false;
40894         }
40895         if(this.maxValue && time > this.maxValue.getTime()){
40896             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40897             return false;
40898         }
40899         /*if(this.disabledDays){
40900             var day = value.getDay();
40901             for(var i = 0; i < this.disabledDays.length; i++) {
40902                 if(day === this.disabledDays[i]){
40903                     this.markInvalid(this.disabledDaysText);
40904                     return false;
40905                 }
40906             }
40907         }
40908         */
40909         var fvalue = this.formatDate(value);
40910         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40911             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40912             return false;
40913         }
40914         */
40915         return true;
40916     },
40917
40918     // private
40919     // Provides logic to override the default TriggerField.validateBlur which just returns true
40920     validateBlur : function(){
40921         return !this.menu || !this.menu.isVisible();
40922     },
40923
40924     /**
40925      * Returns the current date value of the date field.
40926      * @return {Date} The date value
40927      */
40928     getValue : function(){
40929         
40930         
40931         
40932         return  this.hiddenField ?
40933                 this.hiddenField.value :
40934                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40935     },
40936
40937     /**
40938      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40939      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40940      * (the default format used is "m/d/y").
40941      * <br />Usage:
40942      * <pre><code>
40943 //All of these calls set the same date value (May 4, 2006)
40944
40945 //Pass a date object:
40946 var dt = new Date('5/4/06');
40947 monthField.setValue(dt);
40948
40949 //Pass a date string (default format):
40950 monthField.setValue('5/4/06');
40951
40952 //Pass a date string (custom format):
40953 monthField.format = 'Y-m-d';
40954 monthField.setValue('2006-5-4');
40955 </code></pre>
40956      * @param {String/Date} date The date or valid date string
40957      */
40958     setValue : function(date){
40959         Roo.log('month setValue' + date);
40960         // can only be first of month..
40961         
40962         var val = this.parseDate(date);
40963         
40964         if (this.hiddenField) {
40965             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40966         }
40967         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40968         this.value = this.parseDate(date);
40969     },
40970
40971     // private
40972     parseDate : function(value){
40973         if(!value || value instanceof Date){
40974             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40975             return value;
40976         }
40977         var v = Date.parseDate(value, this.format);
40978         if (!v && this.useIso) {
40979             v = Date.parseDate(value, 'Y-m-d');
40980         }
40981         if (v) {
40982             // 
40983             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40984         }
40985         
40986         
40987         if(!v && this.altFormats){
40988             if(!this.altFormatsArray){
40989                 this.altFormatsArray = this.altFormats.split("|");
40990             }
40991             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40992                 v = Date.parseDate(value, this.altFormatsArray[i]);
40993             }
40994         }
40995         return v;
40996     },
40997
40998     // private
40999     formatDate : function(date, fmt){
41000         return (!date || !(date instanceof Date)) ?
41001                date : date.dateFormat(fmt || this.format);
41002     },
41003
41004     // private
41005     menuListeners : {
41006         select: function(m, d){
41007             this.setValue(d);
41008             this.fireEvent('select', this, d);
41009         },
41010         show : function(){ // retain focus styling
41011             this.onFocus();
41012         },
41013         hide : function(){
41014             this.focus.defer(10, this);
41015             var ml = this.menuListeners;
41016             this.menu.un("select", ml.select,  this);
41017             this.menu.un("show", ml.show,  this);
41018             this.menu.un("hide", ml.hide,  this);
41019         }
41020     },
41021     // private
41022     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41023     onTriggerClick : function(){
41024         if(this.disabled){
41025             return;
41026         }
41027         if(this.menu == null){
41028             this.menu = new Roo.menu.DateMenu();
41029            
41030         }
41031         
41032         Roo.apply(this.menu.picker,  {
41033             
41034             showClear: this.allowBlank,
41035             minDate : this.minValue,
41036             maxDate : this.maxValue,
41037             disabledDatesRE : this.ddMatch,
41038             disabledDatesText : this.disabledDatesText,
41039             
41040             format : this.useIso ? 'Y-m-d' : this.format,
41041             minText : String.format(this.minText, this.formatDate(this.minValue)),
41042             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41043             
41044         });
41045          this.menu.on(Roo.apply({}, this.menuListeners, {
41046             scope:this
41047         }));
41048        
41049         
41050         var m = this.menu;
41051         var p = m.picker;
41052         
41053         // hide month picker get's called when we called by 'before hide';
41054         
41055         var ignorehide = true;
41056         p.hideMonthPicker  = function(disableAnim){
41057             if (ignorehide) {
41058                 return;
41059             }
41060              if(this.monthPicker){
41061                 Roo.log("hideMonthPicker called");
41062                 if(disableAnim === true){
41063                     this.monthPicker.hide();
41064                 }else{
41065                     this.monthPicker.slideOut('t', {duration:.2});
41066                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41067                     p.fireEvent("select", this, this.value);
41068                     m.hide();
41069                 }
41070             }
41071         }
41072         
41073         Roo.log('picker set value');
41074         Roo.log(this.getValue());
41075         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41076         m.show(this.el, 'tl-bl?');
41077         ignorehide  = false;
41078         // this will trigger hideMonthPicker..
41079         
41080         
41081         // hidden the day picker
41082         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41083         
41084         
41085         
41086       
41087         
41088         p.showMonthPicker.defer(100, p);
41089     
41090         
41091        
41092     },
41093
41094     beforeBlur : function(){
41095         var v = this.parseDate(this.getRawValue());
41096         if(v){
41097             this.setValue(v);
41098         }
41099     }
41100
41101     /** @cfg {Boolean} grow @hide */
41102     /** @cfg {Number} growMin @hide */
41103     /** @cfg {Number} growMax @hide */
41104     /**
41105      * @hide
41106      * @method autoSize
41107      */
41108 });/*
41109  * Based on:
41110  * Ext JS Library 1.1.1
41111  * Copyright(c) 2006-2007, Ext JS, LLC.
41112  *
41113  * Originally Released Under LGPL - original licence link has changed is not relivant.
41114  *
41115  * Fork - LGPL
41116  * <script type="text/javascript">
41117  */
41118  
41119
41120 /**
41121  * @class Roo.form.ComboBox
41122  * @extends Roo.form.TriggerField
41123  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41124  * @constructor
41125  * Create a new ComboBox.
41126  * @param {Object} config Configuration options
41127  */
41128 Roo.form.ComboBox = function(config){
41129     Roo.form.ComboBox.superclass.constructor.call(this, config);
41130     this.addEvents({
41131         /**
41132          * @event expand
41133          * Fires when the dropdown list is expanded
41134              * @param {Roo.form.ComboBox} combo This combo box
41135              */
41136         'expand' : true,
41137         /**
41138          * @event collapse
41139          * Fires when the dropdown list is collapsed
41140              * @param {Roo.form.ComboBox} combo This combo box
41141              */
41142         'collapse' : true,
41143         /**
41144          * @event beforeselect
41145          * Fires before a list item is selected. Return false to cancel the selection.
41146              * @param {Roo.form.ComboBox} combo This combo box
41147              * @param {Roo.data.Record} record The data record returned from the underlying store
41148              * @param {Number} index The index of the selected item in the dropdown list
41149              */
41150         'beforeselect' : true,
41151         /**
41152          * @event select
41153          * Fires when a list item is selected
41154              * @param {Roo.form.ComboBox} combo This combo box
41155              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41156              * @param {Number} index The index of the selected item in the dropdown list
41157              */
41158         'select' : true,
41159         /**
41160          * @event beforequery
41161          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41162          * The event object passed has these properties:
41163              * @param {Roo.form.ComboBox} combo This combo box
41164              * @param {String} query The query
41165              * @param {Boolean} forceAll true to force "all" query
41166              * @param {Boolean} cancel true to cancel the query
41167              * @param {Object} e The query event object
41168              */
41169         'beforequery': true,
41170          /**
41171          * @event add
41172          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41173              * @param {Roo.form.ComboBox} combo This combo box
41174              */
41175         'add' : true,
41176         /**
41177          * @event edit
41178          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41179              * @param {Roo.form.ComboBox} combo This combo box
41180              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41181              */
41182         'edit' : true
41183         
41184         
41185     });
41186     if(this.transform){
41187         this.allowDomMove = false;
41188         var s = Roo.getDom(this.transform);
41189         if(!this.hiddenName){
41190             this.hiddenName = s.name;
41191         }
41192         if(!this.store){
41193             this.mode = 'local';
41194             var d = [], opts = s.options;
41195             for(var i = 0, len = opts.length;i < len; i++){
41196                 var o = opts[i];
41197                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41198                 if(o.selected) {
41199                     this.value = value;
41200                 }
41201                 d.push([value, o.text]);
41202             }
41203             this.store = new Roo.data.SimpleStore({
41204                 'id': 0,
41205                 fields: ['value', 'text'],
41206                 data : d
41207             });
41208             this.valueField = 'value';
41209             this.displayField = 'text';
41210         }
41211         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41212         if(!this.lazyRender){
41213             this.target = true;
41214             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41215             s.parentNode.removeChild(s); // remove it
41216             this.render(this.el.parentNode);
41217         }else{
41218             s.parentNode.removeChild(s); // remove it
41219         }
41220
41221     }
41222     if (this.store) {
41223         this.store = Roo.factory(this.store, Roo.data);
41224     }
41225     
41226     this.selectedIndex = -1;
41227     if(this.mode == 'local'){
41228         if(config.queryDelay === undefined){
41229             this.queryDelay = 10;
41230         }
41231         if(config.minChars === undefined){
41232             this.minChars = 0;
41233         }
41234     }
41235 };
41236
41237 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41238     /**
41239      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41240      */
41241     /**
41242      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41243      * rendering into an Roo.Editor, defaults to false)
41244      */
41245     /**
41246      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41247      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41248      */
41249     /**
41250      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41251      */
41252     /**
41253      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41254      * the dropdown list (defaults to undefined, with no header element)
41255      */
41256
41257      /**
41258      * @cfg {String/Roo.Template} tpl The template to use to render the output
41259      */
41260      
41261     // private
41262     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41263     /**
41264      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41265      */
41266     listWidth: undefined,
41267     /**
41268      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41269      * mode = 'remote' or 'text' if mode = 'local')
41270      */
41271     displayField: undefined,
41272     /**
41273      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41274      * mode = 'remote' or 'value' if mode = 'local'). 
41275      * Note: use of a valueField requires the user make a selection
41276      * in order for a value to be mapped.
41277      */
41278     valueField: undefined,
41279     
41280     
41281     /**
41282      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41283      * field's data value (defaults to the underlying DOM element's name)
41284      */
41285     hiddenName: undefined,
41286     /**
41287      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41288      */
41289     listClass: '',
41290     /**
41291      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41292      */
41293     selectedClass: 'x-combo-selected',
41294     /**
41295      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41296      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41297      * which displays a downward arrow icon).
41298      */
41299     triggerClass : 'x-form-arrow-trigger',
41300     /**
41301      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41302      */
41303     shadow:'sides',
41304     /**
41305      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41306      * anchor positions (defaults to 'tl-bl')
41307      */
41308     listAlign: 'tl-bl?',
41309     /**
41310      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41311      */
41312     maxHeight: 300,
41313     /**
41314      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41315      * query specified by the allQuery config option (defaults to 'query')
41316      */
41317     triggerAction: 'query',
41318     /**
41319      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41320      * (defaults to 4, does not apply if editable = false)
41321      */
41322     minChars : 4,
41323     /**
41324      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41325      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41326      */
41327     typeAhead: false,
41328     /**
41329      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41330      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41331      */
41332     queryDelay: 500,
41333     /**
41334      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41335      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41336      */
41337     pageSize: 0,
41338     /**
41339      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41340      * when editable = true (defaults to false)
41341      */
41342     selectOnFocus:false,
41343     /**
41344      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41345      */
41346     queryParam: 'query',
41347     /**
41348      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41349      * when mode = 'remote' (defaults to 'Loading...')
41350      */
41351     loadingText: 'Loading...',
41352     /**
41353      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41354      */
41355     resizable: false,
41356     /**
41357      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41358      */
41359     handleHeight : 8,
41360     /**
41361      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41362      * traditional select (defaults to true)
41363      */
41364     editable: true,
41365     /**
41366      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41367      */
41368     allQuery: '',
41369     /**
41370      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41371      */
41372     mode: 'remote',
41373     /**
41374      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41375      * listWidth has a higher value)
41376      */
41377     minListWidth : 70,
41378     /**
41379      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41380      * allow the user to set arbitrary text into the field (defaults to false)
41381      */
41382     forceSelection:false,
41383     /**
41384      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41385      * if typeAhead = true (defaults to 250)
41386      */
41387     typeAheadDelay : 250,
41388     /**
41389      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41390      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41391      */
41392     valueNotFoundText : undefined,
41393     /**
41394      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41395      */
41396     blockFocus : false,
41397     
41398     /**
41399      * @cfg {Boolean} disableClear Disable showing of clear button.
41400      */
41401     disableClear : false,
41402     /**
41403      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41404      */
41405     alwaysQuery : false,
41406     
41407     //private
41408     addicon : false,
41409     editicon: false,
41410     
41411     // element that contains real text value.. (when hidden is used..)
41412      
41413     // private
41414     onRender : function(ct, position)
41415     {
41416         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41417         
41418         if(this.hiddenName){
41419             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41420                     'before', true);
41421             this.hiddenField.value =
41422                 this.hiddenValue !== undefined ? this.hiddenValue :
41423                 this.value !== undefined ? this.value : '';
41424
41425             // prevent input submission
41426             this.el.dom.removeAttribute('name');
41427              
41428              
41429         }
41430         
41431         if(Roo.isGecko){
41432             this.el.dom.setAttribute('autocomplete', 'off');
41433         }
41434
41435         var cls = 'x-combo-list';
41436
41437         this.list = new Roo.Layer({
41438             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41439         });
41440
41441         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41442         this.list.setWidth(lw);
41443         this.list.swallowEvent('mousewheel');
41444         this.assetHeight = 0;
41445
41446         if(this.title){
41447             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41448             this.assetHeight += this.header.getHeight();
41449         }
41450
41451         this.innerList = this.list.createChild({cls:cls+'-inner'});
41452         this.innerList.on('mouseover', this.onViewOver, this);
41453         this.innerList.on('mousemove', this.onViewMove, this);
41454         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41455         
41456         if(this.allowBlank && !this.pageSize && !this.disableClear){
41457             this.footer = this.list.createChild({cls:cls+'-ft'});
41458             this.pageTb = new Roo.Toolbar(this.footer);
41459            
41460         }
41461         if(this.pageSize){
41462             this.footer = this.list.createChild({cls:cls+'-ft'});
41463             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41464                     {pageSize: this.pageSize});
41465             
41466         }
41467         
41468         if (this.pageTb && this.allowBlank && !this.disableClear) {
41469             var _this = this;
41470             this.pageTb.add(new Roo.Toolbar.Fill(), {
41471                 cls: 'x-btn-icon x-btn-clear',
41472                 text: '&#160;',
41473                 handler: function()
41474                 {
41475                     _this.collapse();
41476                     _this.clearValue();
41477                     _this.onSelect(false, -1);
41478                 }
41479             });
41480         }
41481         if (this.footer) {
41482             this.assetHeight += this.footer.getHeight();
41483         }
41484         
41485
41486         if(!this.tpl){
41487             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41488         }
41489
41490         this.view = new Roo.View(this.innerList, this.tpl, {
41491             singleSelect:true,
41492             store: this.store,
41493             selectedClass: this.selectedClass
41494         });
41495
41496         this.view.on('click', this.onViewClick, this);
41497
41498         this.store.on('beforeload', this.onBeforeLoad, this);
41499         this.store.on('load', this.onLoad, this);
41500         this.store.on('loadexception', this.onLoadException, this);
41501
41502         if(this.resizable){
41503             this.resizer = new Roo.Resizable(this.list,  {
41504                pinned:true, handles:'se'
41505             });
41506             this.resizer.on('resize', function(r, w, h){
41507                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41508                 this.listWidth = w;
41509                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41510                 this.restrictHeight();
41511             }, this);
41512             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41513         }
41514         if(!this.editable){
41515             this.editable = true;
41516             this.setEditable(false);
41517         }  
41518         
41519         
41520         if (typeof(this.events.add.listeners) != 'undefined') {
41521             
41522             this.addicon = this.wrap.createChild(
41523                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41524        
41525             this.addicon.on('click', function(e) {
41526                 this.fireEvent('add', this);
41527             }, this);
41528         }
41529         if (typeof(this.events.edit.listeners) != 'undefined') {
41530             
41531             this.editicon = this.wrap.createChild(
41532                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41533             if (this.addicon) {
41534                 this.editicon.setStyle('margin-left', '40px');
41535             }
41536             this.editicon.on('click', function(e) {
41537                 
41538                 // we fire even  if inothing is selected..
41539                 this.fireEvent('edit', this, this.lastData );
41540                 
41541             }, this);
41542         }
41543         
41544         
41545         
41546     },
41547
41548     // private
41549     initEvents : function(){
41550         Roo.form.ComboBox.superclass.initEvents.call(this);
41551
41552         this.keyNav = new Roo.KeyNav(this.el, {
41553             "up" : function(e){
41554                 this.inKeyMode = true;
41555                 this.selectPrev();
41556             },
41557
41558             "down" : function(e){
41559                 if(!this.isExpanded()){
41560                     this.onTriggerClick();
41561                 }else{
41562                     this.inKeyMode = true;
41563                     this.selectNext();
41564                 }
41565             },
41566
41567             "enter" : function(e){
41568                 this.onViewClick();
41569                 //return true;
41570             },
41571
41572             "esc" : function(e){
41573                 this.collapse();
41574             },
41575
41576             "tab" : function(e){
41577                 this.onViewClick(false);
41578                 this.fireEvent("specialkey", this, e);
41579                 return true;
41580             },
41581
41582             scope : this,
41583
41584             doRelay : function(foo, bar, hname){
41585                 if(hname == 'down' || this.scope.isExpanded()){
41586                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41587                 }
41588                 return true;
41589             },
41590
41591             forceKeyDown: true
41592         });
41593         this.queryDelay = Math.max(this.queryDelay || 10,
41594                 this.mode == 'local' ? 10 : 250);
41595         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41596         if(this.typeAhead){
41597             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41598         }
41599         if(this.editable !== false){
41600             this.el.on("keyup", this.onKeyUp, this);
41601         }
41602         if(this.forceSelection){
41603             this.on('blur', this.doForce, this);
41604         }
41605     },
41606
41607     onDestroy : function(){
41608         if(this.view){
41609             this.view.setStore(null);
41610             this.view.el.removeAllListeners();
41611             this.view.el.remove();
41612             this.view.purgeListeners();
41613         }
41614         if(this.list){
41615             this.list.destroy();
41616         }
41617         if(this.store){
41618             this.store.un('beforeload', this.onBeforeLoad, this);
41619             this.store.un('load', this.onLoad, this);
41620             this.store.un('loadexception', this.onLoadException, this);
41621         }
41622         Roo.form.ComboBox.superclass.onDestroy.call(this);
41623     },
41624
41625     // private
41626     fireKey : function(e){
41627         if(e.isNavKeyPress() && !this.list.isVisible()){
41628             this.fireEvent("specialkey", this, e);
41629         }
41630     },
41631
41632     // private
41633     onResize: function(w, h){
41634         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41635         
41636         if(typeof w != 'number'){
41637             // we do not handle it!?!?
41638             return;
41639         }
41640         var tw = this.trigger.getWidth();
41641         tw += this.addicon ? this.addicon.getWidth() : 0;
41642         tw += this.editicon ? this.editicon.getWidth() : 0;
41643         var x = w - tw;
41644         this.el.setWidth( this.adjustWidth('input', x));
41645             
41646         this.trigger.setStyle('left', x+'px');
41647         
41648         if(this.list && this.listWidth === undefined){
41649             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41650             this.list.setWidth(lw);
41651             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41652         }
41653         
41654     
41655         
41656     },
41657
41658     /**
41659      * Allow or prevent the user from directly editing the field text.  If false is passed,
41660      * the user will only be able to select from the items defined in the dropdown list.  This method
41661      * is the runtime equivalent of setting the 'editable' config option at config time.
41662      * @param {Boolean} value True to allow the user to directly edit the field text
41663      */
41664     setEditable : function(value){
41665         if(value == this.editable){
41666             return;
41667         }
41668         this.editable = value;
41669         if(!value){
41670             this.el.dom.setAttribute('readOnly', true);
41671             this.el.on('mousedown', this.onTriggerClick,  this);
41672             this.el.addClass('x-combo-noedit');
41673         }else{
41674             this.el.dom.setAttribute('readOnly', false);
41675             this.el.un('mousedown', this.onTriggerClick,  this);
41676             this.el.removeClass('x-combo-noedit');
41677         }
41678     },
41679
41680     // private
41681     onBeforeLoad : function(){
41682         if(!this.hasFocus){
41683             return;
41684         }
41685         this.innerList.update(this.loadingText ?
41686                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41687         this.restrictHeight();
41688         this.selectedIndex = -1;
41689     },
41690
41691     // private
41692     onLoad : function(){
41693         if(!this.hasFocus){
41694             return;
41695         }
41696         if(this.store.getCount() > 0){
41697             this.expand();
41698             this.restrictHeight();
41699             if(this.lastQuery == this.allQuery){
41700                 if(this.editable){
41701                     this.el.dom.select();
41702                 }
41703                 if(!this.selectByValue(this.value, true)){
41704                     this.select(0, true);
41705                 }
41706             }else{
41707                 this.selectNext();
41708                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41709                     this.taTask.delay(this.typeAheadDelay);
41710                 }
41711             }
41712         }else{
41713             this.onEmptyResults();
41714         }
41715         //this.el.focus();
41716     },
41717     // private
41718     onLoadException : function()
41719     {
41720         this.collapse();
41721         Roo.log(this.store.reader.jsonData);
41722         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41723             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41724         }
41725         
41726         
41727     },
41728     // private
41729     onTypeAhead : function(){
41730         if(this.store.getCount() > 0){
41731             var r = this.store.getAt(0);
41732             var newValue = r.data[this.displayField];
41733             var len = newValue.length;
41734             var selStart = this.getRawValue().length;
41735             if(selStart != len){
41736                 this.setRawValue(newValue);
41737                 this.selectText(selStart, newValue.length);
41738             }
41739         }
41740     },
41741
41742     // private
41743     onSelect : function(record, index){
41744         if(this.fireEvent('beforeselect', this, record, index) !== false){
41745             this.setFromData(index > -1 ? record.data : false);
41746             this.collapse();
41747             this.fireEvent('select', this, record, index);
41748         }
41749     },
41750
41751     /**
41752      * Returns the currently selected field value or empty string if no value is set.
41753      * @return {String} value The selected value
41754      */
41755     getValue : function(){
41756         if(this.valueField){
41757             return typeof this.value != 'undefined' ? this.value : '';
41758         }
41759         return Roo.form.ComboBox.superclass.getValue.call(this);
41760     },
41761
41762     /**
41763      * Clears any text/value currently set in the field
41764      */
41765     clearValue : function(){
41766         if(this.hiddenField){
41767             this.hiddenField.value = '';
41768         }
41769         this.value = '';
41770         this.setRawValue('');
41771         this.lastSelectionText = '';
41772         
41773     },
41774
41775     /**
41776      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41777      * will be displayed in the field.  If the value does not match the data value of an existing item,
41778      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41779      * Otherwise the field will be blank (although the value will still be set).
41780      * @param {String} value The value to match
41781      */
41782     setValue : function(v){
41783         var text = v;
41784         if(this.valueField){
41785             var r = this.findRecord(this.valueField, v);
41786             if(r){
41787                 text = r.data[this.displayField];
41788             }else if(this.valueNotFoundText !== undefined){
41789                 text = this.valueNotFoundText;
41790             }
41791         }
41792         this.lastSelectionText = text;
41793         if(this.hiddenField){
41794             this.hiddenField.value = v;
41795         }
41796         Roo.form.ComboBox.superclass.setValue.call(this, text);
41797         this.value = v;
41798     },
41799     /**
41800      * @property {Object} the last set data for the element
41801      */
41802     
41803     lastData : false,
41804     /**
41805      * Sets the value of the field based on a object which is related to the record format for the store.
41806      * @param {Object} value the value to set as. or false on reset?
41807      */
41808     setFromData : function(o){
41809         var dv = ''; // display value
41810         var vv = ''; // value value..
41811         this.lastData = o;
41812         if (this.displayField) {
41813             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41814         } else {
41815             // this is an error condition!!!
41816             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41817         }
41818         
41819         if(this.valueField){
41820             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41821         }
41822         if(this.hiddenField){
41823             this.hiddenField.value = vv;
41824             
41825             this.lastSelectionText = dv;
41826             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41827             this.value = vv;
41828             return;
41829         }
41830         // no hidden field.. - we store the value in 'value', but still display
41831         // display field!!!!
41832         this.lastSelectionText = dv;
41833         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41834         this.value = vv;
41835         
41836         
41837     },
41838     // private
41839     reset : function(){
41840         // overridden so that last data is reset..
41841         this.setValue(this.resetValue);
41842         this.originalValue = this.getValue();
41843         this.clearInvalid();
41844         this.lastData = false;
41845         if (this.view) {
41846             this.view.clearSelections();
41847         }
41848     },
41849     // private
41850     findRecord : function(prop, value){
41851         var record;
41852         if(this.store.getCount() > 0){
41853             this.store.each(function(r){
41854                 if(r.data[prop] == value){
41855                     record = r;
41856                     return false;
41857                 }
41858                 return true;
41859             });
41860         }
41861         return record;
41862     },
41863     
41864     getName: function()
41865     {
41866         // returns hidden if it's set..
41867         if (!this.rendered) {return ''};
41868         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41869         
41870     },
41871     // private
41872     onViewMove : function(e, t){
41873         this.inKeyMode = false;
41874     },
41875
41876     // private
41877     onViewOver : function(e, t){
41878         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41879             return;
41880         }
41881         var item = this.view.findItemFromChild(t);
41882         if(item){
41883             var index = this.view.indexOf(item);
41884             this.select(index, false);
41885         }
41886     },
41887
41888     // private
41889     onViewClick : function(doFocus)
41890     {
41891         var index = this.view.getSelectedIndexes()[0];
41892         var r = this.store.getAt(index);
41893         if(r){
41894             this.onSelect(r, index);
41895         }
41896         if(doFocus !== false && !this.blockFocus){
41897             this.el.focus();
41898         }
41899     },
41900
41901     // private
41902     restrictHeight : function(){
41903         this.innerList.dom.style.height = '';
41904         var inner = this.innerList.dom;
41905         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41906         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41907         this.list.beginUpdate();
41908         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41909         this.list.alignTo(this.el, this.listAlign);
41910         this.list.endUpdate();
41911     },
41912
41913     // private
41914     onEmptyResults : function(){
41915         this.collapse();
41916     },
41917
41918     /**
41919      * Returns true if the dropdown list is expanded, else false.
41920      */
41921     isExpanded : function(){
41922         return this.list.isVisible();
41923     },
41924
41925     /**
41926      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41927      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41928      * @param {String} value The data value of the item to select
41929      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41930      * selected item if it is not currently in view (defaults to true)
41931      * @return {Boolean} True if the value matched an item in the list, else false
41932      */
41933     selectByValue : function(v, scrollIntoView){
41934         if(v !== undefined && v !== null){
41935             var r = this.findRecord(this.valueField || this.displayField, v);
41936             if(r){
41937                 this.select(this.store.indexOf(r), scrollIntoView);
41938                 return true;
41939             }
41940         }
41941         return false;
41942     },
41943
41944     /**
41945      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41946      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41947      * @param {Number} index The zero-based index of the list item to select
41948      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41949      * selected item if it is not currently in view (defaults to true)
41950      */
41951     select : function(index, scrollIntoView){
41952         this.selectedIndex = index;
41953         this.view.select(index);
41954         if(scrollIntoView !== false){
41955             var el = this.view.getNode(index);
41956             if(el){
41957                 this.innerList.scrollChildIntoView(el, false);
41958             }
41959         }
41960     },
41961
41962     // private
41963     selectNext : function(){
41964         var ct = this.store.getCount();
41965         if(ct > 0){
41966             if(this.selectedIndex == -1){
41967                 this.select(0);
41968             }else if(this.selectedIndex < ct-1){
41969                 this.select(this.selectedIndex+1);
41970             }
41971         }
41972     },
41973
41974     // private
41975     selectPrev : function(){
41976         var ct = this.store.getCount();
41977         if(ct > 0){
41978             if(this.selectedIndex == -1){
41979                 this.select(0);
41980             }else if(this.selectedIndex != 0){
41981                 this.select(this.selectedIndex-1);
41982             }
41983         }
41984     },
41985
41986     // private
41987     onKeyUp : function(e){
41988         if(this.editable !== false && !e.isSpecialKey()){
41989             this.lastKey = e.getKey();
41990             this.dqTask.delay(this.queryDelay);
41991         }
41992     },
41993
41994     // private
41995     validateBlur : function(){
41996         return !this.list || !this.list.isVisible();   
41997     },
41998
41999     // private
42000     initQuery : function(){
42001         this.doQuery(this.getRawValue());
42002     },
42003
42004     // private
42005     doForce : function(){
42006         if(this.el.dom.value.length > 0){
42007             this.el.dom.value =
42008                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
42009              
42010         }
42011     },
42012
42013     /**
42014      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42015      * query allowing the query action to be canceled if needed.
42016      * @param {String} query The SQL query to execute
42017      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42018      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42019      * saved in the current store (defaults to false)
42020      */
42021     doQuery : function(q, forceAll){
42022         if(q === undefined || q === null){
42023             q = '';
42024         }
42025         var qe = {
42026             query: q,
42027             forceAll: forceAll,
42028             combo: this,
42029             cancel:false
42030         };
42031         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42032             return false;
42033         }
42034         q = qe.query;
42035         forceAll = qe.forceAll;
42036         if(forceAll === true || (q.length >= this.minChars)){
42037             if(this.lastQuery != q || this.alwaysQuery){
42038                 this.lastQuery = q;
42039                 if(this.mode == 'local'){
42040                     this.selectedIndex = -1;
42041                     if(forceAll){
42042                         this.store.clearFilter();
42043                     }else{
42044                         this.store.filter(this.displayField, q);
42045                     }
42046                     this.onLoad();
42047                 }else{
42048                     this.store.baseParams[this.queryParam] = q;
42049                     this.store.load({
42050                         params: this.getParams(q)
42051                     });
42052                     this.expand();
42053                 }
42054             }else{
42055                 this.selectedIndex = -1;
42056                 this.onLoad();   
42057             }
42058         }
42059     },
42060
42061     // private
42062     getParams : function(q){
42063         var p = {};
42064         //p[this.queryParam] = q;
42065         if(this.pageSize){
42066             p.start = 0;
42067             p.limit = this.pageSize;
42068         }
42069         return p;
42070     },
42071
42072     /**
42073      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42074      */
42075     collapse : function(){
42076         if(!this.isExpanded()){
42077             return;
42078         }
42079         this.list.hide();
42080         Roo.get(document).un('mousedown', this.collapseIf, this);
42081         Roo.get(document).un('mousewheel', this.collapseIf, this);
42082         if (!this.editable) {
42083             Roo.get(document).un('keydown', this.listKeyPress, this);
42084         }
42085         this.fireEvent('collapse', this);
42086     },
42087
42088     // private
42089     collapseIf : function(e){
42090         if(!e.within(this.wrap) && !e.within(this.list)){
42091             this.collapse();
42092         }
42093     },
42094
42095     /**
42096      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42097      */
42098     expand : function(){
42099         if(this.isExpanded() || !this.hasFocus){
42100             return;
42101         }
42102         this.list.alignTo(this.el, this.listAlign);
42103         this.list.show();
42104         Roo.get(document).on('mousedown', this.collapseIf, this);
42105         Roo.get(document).on('mousewheel', this.collapseIf, this);
42106         if (!this.editable) {
42107             Roo.get(document).on('keydown', this.listKeyPress, this);
42108         }
42109         
42110         this.fireEvent('expand', this);
42111     },
42112
42113     // private
42114     // Implements the default empty TriggerField.onTriggerClick function
42115     onTriggerClick : function(){
42116         if(this.disabled){
42117             return;
42118         }
42119         if(this.isExpanded()){
42120             this.collapse();
42121             if (!this.blockFocus) {
42122                 this.el.focus();
42123             }
42124             
42125         }else {
42126             this.hasFocus = true;
42127             if(this.triggerAction == 'all') {
42128                 this.doQuery(this.allQuery, true);
42129             } else {
42130                 this.doQuery(this.getRawValue());
42131             }
42132             if (!this.blockFocus) {
42133                 this.el.focus();
42134             }
42135         }
42136     },
42137     listKeyPress : function(e)
42138     {
42139         //Roo.log('listkeypress');
42140         // scroll to first matching element based on key pres..
42141         if (e.isSpecialKey()) {
42142             return false;
42143         }
42144         var k = String.fromCharCode(e.getKey()).toUpperCase();
42145         //Roo.log(k);
42146         var match  = false;
42147         var csel = this.view.getSelectedNodes();
42148         var cselitem = false;
42149         if (csel.length) {
42150             var ix = this.view.indexOf(csel[0]);
42151             cselitem  = this.store.getAt(ix);
42152             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42153                 cselitem = false;
42154             }
42155             
42156         }
42157         
42158         this.store.each(function(v) { 
42159             if (cselitem) {
42160                 // start at existing selection.
42161                 if (cselitem.id == v.id) {
42162                     cselitem = false;
42163                 }
42164                 return;
42165             }
42166                 
42167             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42168                 match = this.store.indexOf(v);
42169                 return false;
42170             }
42171         }, this);
42172         
42173         if (match === false) {
42174             return true; // no more action?
42175         }
42176         // scroll to?
42177         this.view.select(match);
42178         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42179         sn.scrollIntoView(sn.dom.parentNode, false);
42180     } 
42181
42182     /** 
42183     * @cfg {Boolean} grow 
42184     * @hide 
42185     */
42186     /** 
42187     * @cfg {Number} growMin 
42188     * @hide 
42189     */
42190     /** 
42191     * @cfg {Number} growMax 
42192     * @hide 
42193     */
42194     /**
42195      * @hide
42196      * @method autoSize
42197      */
42198 });/*
42199  * Copyright(c) 2010-2012, Roo J Solutions Limited
42200  *
42201  * Licence LGPL
42202  *
42203  */
42204
42205 /**
42206  * @class Roo.form.ComboBoxArray
42207  * @extends Roo.form.TextField
42208  * A facebook style adder... for lists of email / people / countries  etc...
42209  * pick multiple items from a combo box, and shows each one.
42210  *
42211  *  Fred [x]  Brian [x]  [Pick another |v]
42212  *
42213  *
42214  *  For this to work: it needs various extra information
42215  *    - normal combo problay has
42216  *      name, hiddenName
42217  *    + displayField, valueField
42218  *
42219  *    For our purpose...
42220  *
42221  *
42222  *   If we change from 'extends' to wrapping...
42223  *   
42224  *  
42225  *
42226  
42227  
42228  * @constructor
42229  * Create a new ComboBoxArray.
42230  * @param {Object} config Configuration options
42231  */
42232  
42233
42234 Roo.form.ComboBoxArray = function(config)
42235 {
42236     this.addEvents({
42237         /**
42238          * @event beforeremove
42239          * Fires before remove the value from the list
42240              * @param {Roo.form.ComboBoxArray} _self This combo box array
42241              * @param {Roo.form.ComboBoxArray.Item} item removed item
42242              */
42243         'beforeremove' : true,
42244         /**
42245          * @event remove
42246          * Fires when remove the value from the list
42247              * @param {Roo.form.ComboBoxArray} _self This combo box array
42248              * @param {Roo.form.ComboBoxArray.Item} item removed item
42249              */
42250         'remove' : true
42251         
42252         
42253     });
42254     
42255     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42256     
42257     this.items = new Roo.util.MixedCollection(false);
42258     
42259     // construct the child combo...
42260     
42261     
42262     
42263     
42264    
42265     
42266 }
42267
42268  
42269 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42270
42271     /**
42272      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42273      */
42274     
42275     lastData : false,
42276     
42277     // behavies liek a hiddne field
42278     inputType:      'hidden',
42279     /**
42280      * @cfg {Number} width The width of the box that displays the selected element
42281      */ 
42282     width:          300,
42283
42284     
42285     
42286     /**
42287      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42288      */
42289     name : false,
42290     /**
42291      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42292      */
42293     hiddenName : false,
42294     
42295     
42296     // private the array of items that are displayed..
42297     items  : false,
42298     // private - the hidden field el.
42299     hiddenEl : false,
42300     // private - the filed el..
42301     el : false,
42302     
42303     //validateValue : function() { return true; }, // all values are ok!
42304     //onAddClick: function() { },
42305     
42306     onRender : function(ct, position) 
42307     {
42308         
42309         // create the standard hidden element
42310         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42311         
42312         
42313         // give fake names to child combo;
42314         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42315         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42316         
42317         this.combo = Roo.factory(this.combo, Roo.form);
42318         this.combo.onRender(ct, position);
42319         if (typeof(this.combo.width) != 'undefined') {
42320             this.combo.onResize(this.combo.width,0);
42321         }
42322         
42323         this.combo.initEvents();
42324         
42325         // assigned so form know we need to do this..
42326         this.store          = this.combo.store;
42327         this.valueField     = this.combo.valueField;
42328         this.displayField   = this.combo.displayField ;
42329         
42330         
42331         this.combo.wrap.addClass('x-cbarray-grp');
42332         
42333         var cbwrap = this.combo.wrap.createChild(
42334             {tag: 'div', cls: 'x-cbarray-cb'},
42335             this.combo.el.dom
42336         );
42337         
42338              
42339         this.hiddenEl = this.combo.wrap.createChild({
42340             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42341         });
42342         this.el = this.combo.wrap.createChild({
42343             tag: 'input',  type:'hidden' , name: this.name, value : ''
42344         });
42345          //   this.el.dom.removeAttribute("name");
42346         
42347         
42348         this.outerWrap = this.combo.wrap;
42349         this.wrap = cbwrap;
42350         
42351         this.outerWrap.setWidth(this.width);
42352         this.outerWrap.dom.removeChild(this.el.dom);
42353         
42354         this.wrap.dom.appendChild(this.el.dom);
42355         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42356         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42357         
42358         this.combo.trigger.setStyle('position','relative');
42359         this.combo.trigger.setStyle('left', '0px');
42360         this.combo.trigger.setStyle('top', '2px');
42361         
42362         this.combo.el.setStyle('vertical-align', 'text-bottom');
42363         
42364         //this.trigger.setStyle('vertical-align', 'top');
42365         
42366         // this should use the code from combo really... on('add' ....)
42367         if (this.adder) {
42368             
42369         
42370             this.adder = this.outerWrap.createChild(
42371                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42372             var _t = this;
42373             this.adder.on('click', function(e) {
42374                 _t.fireEvent('adderclick', this, e);
42375             }, _t);
42376         }
42377         //var _t = this;
42378         //this.adder.on('click', this.onAddClick, _t);
42379         
42380         
42381         this.combo.on('select', function(cb, rec, ix) {
42382             this.addItem(rec.data);
42383             
42384             cb.setValue('');
42385             cb.el.dom.value = '';
42386             //cb.lastData = rec.data;
42387             // add to list
42388             
42389         }, this);
42390         
42391         
42392     },
42393     
42394     
42395     getName: function()
42396     {
42397         // returns hidden if it's set..
42398         if (!this.rendered) {return ''};
42399         return  this.hiddenName ? this.hiddenName : this.name;
42400         
42401     },
42402     
42403     
42404     onResize: function(w, h){
42405         
42406         return;
42407         // not sure if this is needed..
42408         //this.combo.onResize(w,h);
42409         
42410         if(typeof w != 'number'){
42411             // we do not handle it!?!?
42412             return;
42413         }
42414         var tw = this.combo.trigger.getWidth();
42415         tw += this.addicon ? this.addicon.getWidth() : 0;
42416         tw += this.editicon ? this.editicon.getWidth() : 0;
42417         var x = w - tw;
42418         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42419             
42420         this.combo.trigger.setStyle('left', '0px');
42421         
42422         if(this.list && this.listWidth === undefined){
42423             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42424             this.list.setWidth(lw);
42425             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42426         }
42427         
42428     
42429         
42430     },
42431     
42432     addItem: function(rec)
42433     {
42434         var valueField = this.combo.valueField;
42435         var displayField = this.combo.displayField;
42436         
42437         if (this.items.indexOfKey(rec[valueField]) > -1) {
42438             //console.log("GOT " + rec.data.id);
42439             return;
42440         }
42441         
42442         var x = new Roo.form.ComboBoxArray.Item({
42443             //id : rec[this.idField],
42444             data : rec,
42445             displayField : displayField ,
42446             tipField : displayField ,
42447             cb : this
42448         });
42449         // use the 
42450         this.items.add(rec[valueField],x);
42451         // add it before the element..
42452         this.updateHiddenEl();
42453         x.render(this.outerWrap, this.wrap.dom);
42454         // add the image handler..
42455     },
42456     
42457     updateHiddenEl : function()
42458     {
42459         this.validate();
42460         if (!this.hiddenEl) {
42461             return;
42462         }
42463         var ar = [];
42464         var idField = this.combo.valueField;
42465         
42466         this.items.each(function(f) {
42467             ar.push(f.data[idField]);
42468         });
42469         this.hiddenEl.dom.value = ar.join(',');
42470         this.validate();
42471     },
42472     
42473     reset : function()
42474     {
42475         this.items.clear();
42476         
42477         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42478            el.remove();
42479         });
42480         
42481         this.el.dom.value = '';
42482         if (this.hiddenEl) {
42483             this.hiddenEl.dom.value = '';
42484         }
42485         
42486     },
42487     getValue: function()
42488     {
42489         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42490     },
42491     setValue: function(v) // not a valid action - must use addItems..
42492     {
42493         
42494         this.reset();
42495          
42496         if (this.store.isLocal && (typeof(v) == 'string')) {
42497             // then we can use the store to find the values..
42498             // comma seperated at present.. this needs to allow JSON based encoding..
42499             this.hiddenEl.value  = v;
42500             var v_ar = [];
42501             Roo.each(v.split(','), function(k) {
42502                 Roo.log("CHECK " + this.valueField + ',' + k);
42503                 var li = this.store.query(this.valueField, k);
42504                 if (!li.length) {
42505                     return;
42506                 }
42507                 var add = {};
42508                 add[this.valueField] = k;
42509                 add[this.displayField] = li.item(0).data[this.displayField];
42510                 
42511                 this.addItem(add);
42512             }, this) 
42513              
42514         }
42515         if (typeof(v) == 'object' ) {
42516             // then let's assume it's an array of objects..
42517             Roo.each(v, function(l) {
42518                 this.addItem(l);
42519             }, this);
42520              
42521         }
42522         
42523         
42524     },
42525     setFromData: function(v)
42526     {
42527         // this recieves an object, if setValues is called.
42528         this.reset();
42529         this.el.dom.value = v[this.displayField];
42530         this.hiddenEl.dom.value = v[this.valueField];
42531         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42532             return;
42533         }
42534         var kv = v[this.valueField];
42535         var dv = v[this.displayField];
42536         kv = typeof(kv) != 'string' ? '' : kv;
42537         dv = typeof(dv) != 'string' ? '' : dv;
42538         
42539         
42540         var keys = kv.split(',');
42541         var display = dv.split(',');
42542         for (var i = 0 ; i < keys.length; i++) {
42543             
42544             add = {};
42545             add[this.valueField] = keys[i];
42546             add[this.displayField] = display[i];
42547             this.addItem(add);
42548         }
42549       
42550         
42551     },
42552     
42553     /**
42554      * Validates the combox array value
42555      * @return {Boolean} True if the value is valid, else false
42556      */
42557     validate : function(){
42558         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42559             this.clearInvalid();
42560             return true;
42561         }
42562         return false;
42563     },
42564     
42565     validateValue : function(value){
42566         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42567         
42568     },
42569     
42570     /*@
42571      * overide
42572      * 
42573      */
42574     isDirty : function() {
42575         if(this.disabled) {
42576             return false;
42577         }
42578         
42579         try {
42580             var d = Roo.decode(String(this.originalValue));
42581         } catch (e) {
42582             return String(this.getValue()) !== String(this.originalValue);
42583         }
42584         
42585         var originalValue = [];
42586         
42587         for (var i = 0; i < d.length; i++){
42588             originalValue.push(d[i][this.valueField]);
42589         }
42590         
42591         return String(this.getValue()) !== String(originalValue.join(','));
42592         
42593     }
42594     
42595 });
42596
42597
42598
42599 /**
42600  * @class Roo.form.ComboBoxArray.Item
42601  * @extends Roo.BoxComponent
42602  * A selected item in the list
42603  *  Fred [x]  Brian [x]  [Pick another |v]
42604  * 
42605  * @constructor
42606  * Create a new item.
42607  * @param {Object} config Configuration options
42608  */
42609  
42610 Roo.form.ComboBoxArray.Item = function(config) {
42611     config.id = Roo.id();
42612     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42613 }
42614
42615 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42616     data : {},
42617     cb: false,
42618     displayField : false,
42619     tipField : false,
42620     
42621     
42622     defaultAutoCreate : {
42623         tag: 'div',
42624         cls: 'x-cbarray-item',
42625         cn : [ 
42626             { tag: 'div' },
42627             {
42628                 tag: 'img',
42629                 width:16,
42630                 height : 16,
42631                 src : Roo.BLANK_IMAGE_URL ,
42632                 align: 'center'
42633             }
42634         ]
42635         
42636     },
42637     
42638  
42639     onRender : function(ct, position)
42640     {
42641         Roo.form.Field.superclass.onRender.call(this, ct, position);
42642         
42643         if(!this.el){
42644             var cfg = this.getAutoCreate();
42645             this.el = ct.createChild(cfg, position);
42646         }
42647         
42648         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42649         
42650         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42651             this.cb.renderer(this.data) :
42652             String.format('{0}',this.data[this.displayField]);
42653         
42654             
42655         this.el.child('div').dom.setAttribute('qtip',
42656                         String.format('{0}',this.data[this.tipField])
42657         );
42658         
42659         this.el.child('img').on('click', this.remove, this);
42660         
42661     },
42662    
42663     remove : function()
42664     {
42665         if(this.cb.disabled){
42666             return;
42667         }
42668         
42669         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42670             this.cb.items.remove(this);
42671             this.el.child('img').un('click', this.remove, this);
42672             this.el.remove();
42673             this.cb.updateHiddenEl();
42674
42675             this.cb.fireEvent('remove', this.cb, this);
42676         }
42677         
42678     }
42679 });/*
42680  * RooJS Library 1.1.1
42681  * Copyright(c) 2008-2011  Alan Knowles
42682  *
42683  * License - LGPL
42684  */
42685  
42686
42687 /**
42688  * @class Roo.form.ComboNested
42689  * @extends Roo.form.ComboBox
42690  * A combobox for that allows selection of nested items in a list,
42691  * eg.
42692  *
42693  *  Book
42694  *    -> red
42695  *    -> green
42696  *  Table
42697  *    -> square
42698  *      ->red
42699  *      ->green
42700  *    -> rectangle
42701  *      ->green
42702  *      
42703  * 
42704  * @constructor
42705  * Create a new ComboNested
42706  * @param {Object} config Configuration options
42707  */
42708 Roo.form.ComboNested = function(config){
42709     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42710     // should verify some data...
42711     // like
42712     // hiddenName = required..
42713     // displayField = required
42714     // valudField == required
42715     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42716     var _t = this;
42717     Roo.each(req, function(e) {
42718         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42719             throw "Roo.form.ComboNested : missing value for: " + e;
42720         }
42721     });
42722      
42723     
42724 };
42725
42726 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42727    
42728     /*
42729      * @config {Number} max Number of columns to show
42730      */
42731     
42732     maxColumns : 3,
42733    
42734     list : null, // the outermost div..
42735     innerLists : null, // the
42736     views : null,
42737     stores : null,
42738     // private
42739     onRender : function(ct, position)
42740     {
42741         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42742         
42743         if(this.hiddenName){
42744             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42745                     'before', true);
42746             this.hiddenField.value =
42747                 this.hiddenValue !== undefined ? this.hiddenValue :
42748                 this.value !== undefined ? this.value : '';
42749
42750             // prevent input submission
42751             this.el.dom.removeAttribute('name');
42752              
42753              
42754         }
42755         
42756         if(Roo.isGecko){
42757             this.el.dom.setAttribute('autocomplete', 'off');
42758         }
42759
42760         var cls = 'x-combo-list';
42761
42762         this.list = new Roo.Layer({
42763             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42764         });
42765
42766         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42767         this.list.setWidth(lw);
42768         this.list.swallowEvent('mousewheel');
42769         this.assetHeight = 0;
42770
42771         if(this.title){
42772             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42773             this.assetHeight += this.header.getHeight();
42774         }
42775         this.innerLists = [];
42776         this.views = [];
42777         this.stores = [];
42778         for (var i =0 ; i < this.maxColumns; i++) {
42779             this.onRenderList( cls, i);
42780         }
42781         
42782         // always needs footer, as we are going to have an 'OK' button.
42783         this.footer = this.list.createChild({cls:cls+'-ft'});
42784         this.pageTb = new Roo.Toolbar(this.footer);  
42785         var _this = this;
42786         this.pageTb.add(  {
42787             
42788             text: 'Done',
42789             handler: function()
42790             {
42791                 _this.collapse();
42792             }
42793         });
42794         
42795         if ( this.allowBlank && !this.disableClear) {
42796             
42797             this.pageTb.add(new Roo.Toolbar.Fill(), {
42798                 cls: 'x-btn-icon x-btn-clear',
42799                 text: '&#160;',
42800                 handler: function()
42801                 {
42802                     _this.collapse();
42803                     _this.clearValue();
42804                     _this.onSelect(false, -1);
42805                 }
42806             });
42807         }
42808         if (this.footer) {
42809             this.assetHeight += this.footer.getHeight();
42810         }
42811         
42812     },
42813     onRenderList : function (  cls, i)
42814     {
42815         
42816         var lw = Math.floor(
42817                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42818         );
42819         
42820         this.list.setWidth(lw); // default to '1'
42821
42822         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42823         //il.on('mouseover', this.onViewOver, this, { list:  i });
42824         //il.on('mousemove', this.onViewMove, this, { list:  i });
42825         il.setWidth(lw);
42826         il.setStyle({ 'overflow-x' : 'hidden'});
42827
42828         if(!this.tpl){
42829             this.tpl = new Roo.Template({
42830                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42831                 isEmpty: function (value, allValues) {
42832                     //Roo.log(value);
42833                     var dl = typeof(value.data) != 'undefined' ? value.data.length : value.length; ///json is a nested response..
42834                     return dl ? 'has-children' : 'no-children'
42835                 }
42836             });
42837         }
42838         
42839         var store  = this.store;
42840         if (i > 0) {
42841             store  = new Roo.data.SimpleStore({
42842                 //fields : this.store.reader.meta.fields,
42843                 reader : this.store.reader,
42844                 data : [ ]
42845             });
42846         }
42847         this.stores[i]  = store;
42848                 
42849         
42850         
42851         var view = this.views[i] = new Roo.View(
42852             il,
42853             this.tpl,
42854             {
42855                 singleSelect:true,
42856                 store: store,
42857                 selectedClass: this.selectedClass
42858             }
42859         );
42860         view.getEl().setWidth(lw);
42861         view.getEl().setStyle({
42862             position: i < 1 ? 'relative' : 'absolute',
42863             top: 0,
42864             left: (i * lw ) + 'px',
42865             display : i > 0 ? 'none' : 'block'
42866         });
42867         view.on('selectionchange', this.onSelectChange, this, {list : i });
42868         view.on('dblclick', this.onDoubleClick, this, {list : i });
42869         //view.on('click', this.onViewClick, this, { list : i });
42870
42871         store.on('beforeload', this.onBeforeLoad, this);
42872         store.on('load',  this.onLoad, this, { list  : i});
42873         store.on('loadexception', this.onLoadException, this);
42874
42875         // hide the other vies..
42876         
42877         
42878         
42879     },
42880     onResize : function()  {},
42881     
42882     restrictHeight : function()
42883     {
42884         var mh = 0;
42885         Roo.each(this.innerLists, function(il,i) {
42886             var el = this.views[i].getEl();
42887             el.dom.style.height = '';
42888             var inner = el.dom;
42889             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42890             // only adjust heights on other ones..
42891             if (i < 1) {
42892                 
42893                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42894                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42895                 mh = Math.max(el.getHeight(), mh);
42896             }
42897             
42898             
42899         }, this);
42900         
42901         this.list.beginUpdate();
42902         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42903         this.list.alignTo(this.el, this.listAlign);
42904         this.list.endUpdate();
42905         
42906     },
42907      
42908     
42909     // -- store handlers..
42910     // private
42911     onBeforeLoad : function()
42912     {
42913         if(!this.hasFocus){
42914             return;
42915         }
42916         this.innerLists[0].update(this.loadingText ?
42917                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42918         this.restrictHeight();
42919         this.selectedIndex = -1;
42920     },
42921     // private
42922     onLoad : function(a,b,c,d)
42923     {
42924         
42925         if(!this.hasFocus){
42926             return;
42927         }
42928         
42929         if(this.store.getCount() > 0) {
42930             this.expand();
42931             this.restrictHeight();   
42932         } else {
42933             this.onEmptyResults();
42934         }
42935         /*
42936         this.stores[1].loadData([]);
42937         this.stores[2].loadData([]);
42938         this.views
42939         */    
42940     
42941         //this.el.focus();
42942     },
42943     
42944     
42945     // private
42946     onLoadException : function()
42947     {
42948         this.collapse();
42949         Roo.log(this.store.reader.jsonData);
42950         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42951             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42952         }
42953         
42954         
42955     } ,
42956      
42957      
42958
42959     onSelectChange : function (view, sels, opts )
42960     {
42961         var ix = view.getSelectedIndexes();
42962         
42963         
42964         if (opts.list > this.maxColumns - 2) {
42965              
42966             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42967             return;
42968         }
42969         
42970         if (!ix.length) {
42971             this.setFromData({});
42972             var str = this.stores[opts.list+1];
42973             str.loadData( str.reader.readerType == 'json' ? { data : [] } : [] );
42974             return;
42975         }
42976         
42977         var rec = view.store.getAt(ix[0]);
42978         this.setFromData(rec.data);
42979         
42980         var lw = Math.floor(
42981                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42982         );
42983         var data =  typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
42984         var dl = typeof(data.data) != 'undefined' ? data.total : data.length; ///json is a nested response..
42985         this.stores[opts.list+1].loadData( data );
42986         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
42987         this.views[opts.list+1].getEl().setStyle({ display : dl ? 'block' : 'none' });
42988         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
42989         this.list.setWidth(lw * (opts.list + (dl ? 2 : 1))); 
42990     },
42991     onDoubleClick : function()
42992     {
42993         this.collapse(); //??
42994     },
42995     
42996      
42997     
42998     findRecord : function (prop,value)
42999     {
43000         return this.findRecordInStore(this.store, prop,value);
43001     },
43002     
43003      // private
43004     findRecordInStore : function(store, prop, value)
43005     {
43006         var cstore = new Roo.data.SimpleStore({
43007             //fields : this.store.reader.meta.fields, // we need array reader.. for
43008             reader : this.store.reader,
43009             data : [ ]
43010         });
43011         var _this = this;
43012         var record  = false;
43013         if(store.getCount() > 0){
43014            store.each(function(r){
43015                 if(r.data[prop] == value){
43016                     record = r;
43017                     return false;
43018                 }
43019                 if (r.data.cn && r.data.cn.length) {
43020                     cstore.loadData( r.data.cn);
43021                     var cret = _this.findRecordInStore(cstore, prop, value);
43022                     if (cret !== false) {
43023                         record = cret;
43024                         return false;
43025                     }
43026                 }
43027                 
43028                 return true;
43029             });
43030         }
43031         return record;
43032     }
43033     
43034     
43035     
43036     
43037 });/*
43038  * Based on:
43039  * Ext JS Library 1.1.1
43040  * Copyright(c) 2006-2007, Ext JS, LLC.
43041  *
43042  * Originally Released Under LGPL - original licence link has changed is not relivant.
43043  *
43044  * Fork - LGPL
43045  * <script type="text/javascript">
43046  */
43047 /**
43048  * @class Roo.form.Checkbox
43049  * @extends Roo.form.Field
43050  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43051  * @constructor
43052  * Creates a new Checkbox
43053  * @param {Object} config Configuration options
43054  */
43055 Roo.form.Checkbox = function(config){
43056     Roo.form.Checkbox.superclass.constructor.call(this, config);
43057     this.addEvents({
43058         /**
43059          * @event check
43060          * Fires when the checkbox is checked or unchecked.
43061              * @param {Roo.form.Checkbox} this This checkbox
43062              * @param {Boolean} checked The new checked value
43063              */
43064         check : true
43065     });
43066 };
43067
43068 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43069     /**
43070      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43071      */
43072     focusClass : undefined,
43073     /**
43074      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43075      */
43076     fieldClass: "x-form-field",
43077     /**
43078      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43079      */
43080     checked: false,
43081     /**
43082      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43083      * {tag: "input", type: "checkbox", autocomplete: "off"})
43084      */
43085     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43086     /**
43087      * @cfg {String} boxLabel The text that appears beside the checkbox
43088      */
43089     boxLabel : "",
43090     /**
43091      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43092      */  
43093     inputValue : '1',
43094     /**
43095      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43096      */
43097      valueOff: '0', // value when not checked..
43098
43099     actionMode : 'viewEl', 
43100     //
43101     // private
43102     itemCls : 'x-menu-check-item x-form-item',
43103     groupClass : 'x-menu-group-item',
43104     inputType : 'hidden',
43105     
43106     
43107     inSetChecked: false, // check that we are not calling self...
43108     
43109     inputElement: false, // real input element?
43110     basedOn: false, // ????
43111     
43112     isFormField: true, // not sure where this is needed!!!!
43113
43114     onResize : function(){
43115         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43116         if(!this.boxLabel){
43117             this.el.alignTo(this.wrap, 'c-c');
43118         }
43119     },
43120
43121     initEvents : function(){
43122         Roo.form.Checkbox.superclass.initEvents.call(this);
43123         this.el.on("click", this.onClick,  this);
43124         this.el.on("change", this.onClick,  this);
43125     },
43126
43127
43128     getResizeEl : function(){
43129         return this.wrap;
43130     },
43131
43132     getPositionEl : function(){
43133         return this.wrap;
43134     },
43135
43136     // private
43137     onRender : function(ct, position){
43138         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43139         /*
43140         if(this.inputValue !== undefined){
43141             this.el.dom.value = this.inputValue;
43142         }
43143         */
43144         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43145         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43146         var viewEl = this.wrap.createChild({ 
43147             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43148         this.viewEl = viewEl;   
43149         this.wrap.on('click', this.onClick,  this); 
43150         
43151         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43152         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43153         
43154         
43155         
43156         if(this.boxLabel){
43157             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43158         //    viewEl.on('click', this.onClick,  this); 
43159         }
43160         //if(this.checked){
43161             this.setChecked(this.checked);
43162         //}else{
43163             //this.checked = this.el.dom;
43164         //}
43165
43166     },
43167
43168     // private
43169     initValue : Roo.emptyFn,
43170
43171     /**
43172      * Returns the checked state of the checkbox.
43173      * @return {Boolean} True if checked, else false
43174      */
43175     getValue : function(){
43176         if(this.el){
43177             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43178         }
43179         return this.valueOff;
43180         
43181     },
43182
43183         // private
43184     onClick : function(){ 
43185         if (this.disabled) {
43186             return;
43187         }
43188         this.setChecked(!this.checked);
43189
43190         //if(this.el.dom.checked != this.checked){
43191         //    this.setValue(this.el.dom.checked);
43192        // }
43193     },
43194
43195     /**
43196      * Sets the checked state of the checkbox.
43197      * On is always based on a string comparison between inputValue and the param.
43198      * @param {Boolean/String} value - the value to set 
43199      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43200      */
43201     setValue : function(v,suppressEvent){
43202         
43203         
43204         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43205         //if(this.el && this.el.dom){
43206         //    this.el.dom.checked = this.checked;
43207         //    this.el.dom.defaultChecked = this.checked;
43208         //}
43209         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43210         //this.fireEvent("check", this, this.checked);
43211     },
43212     // private..
43213     setChecked : function(state,suppressEvent)
43214     {
43215         if (this.inSetChecked) {
43216             this.checked = state;
43217             return;
43218         }
43219         
43220     
43221         if(this.wrap){
43222             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43223         }
43224         this.checked = state;
43225         if(suppressEvent !== true){
43226             this.fireEvent('check', this, state);
43227         }
43228         this.inSetChecked = true;
43229         this.el.dom.value = state ? this.inputValue : this.valueOff;
43230         this.inSetChecked = false;
43231         
43232     },
43233     // handle setting of hidden value by some other method!!?!?
43234     setFromHidden: function()
43235     {
43236         if(!this.el){
43237             return;
43238         }
43239         //console.log("SET FROM HIDDEN");
43240         //alert('setFrom hidden');
43241         this.setValue(this.el.dom.value);
43242     },
43243     
43244     onDestroy : function()
43245     {
43246         if(this.viewEl){
43247             Roo.get(this.viewEl).remove();
43248         }
43249          
43250         Roo.form.Checkbox.superclass.onDestroy.call(this);
43251     },
43252     
43253     setBoxLabel : function(str)
43254     {
43255         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43256     }
43257
43258 });/*
43259  * Based on:
43260  * Ext JS Library 1.1.1
43261  * Copyright(c) 2006-2007, Ext JS, LLC.
43262  *
43263  * Originally Released Under LGPL - original licence link has changed is not relivant.
43264  *
43265  * Fork - LGPL
43266  * <script type="text/javascript">
43267  */
43268  
43269 /**
43270  * @class Roo.form.Radio
43271  * @extends Roo.form.Checkbox
43272  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43273  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43274  * @constructor
43275  * Creates a new Radio
43276  * @param {Object} config Configuration options
43277  */
43278 Roo.form.Radio = function(){
43279     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43280 };
43281 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43282     inputType: 'radio',
43283
43284     /**
43285      * If this radio is part of a group, it will return the selected value
43286      * @return {String}
43287      */
43288     getGroupValue : function(){
43289         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43290     },
43291     
43292     
43293     onRender : function(ct, position){
43294         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43295         
43296         if(this.inputValue !== undefined){
43297             this.el.dom.value = this.inputValue;
43298         }
43299          
43300         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43301         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43302         //var viewEl = this.wrap.createChild({ 
43303         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43304         //this.viewEl = viewEl;   
43305         //this.wrap.on('click', this.onClick,  this); 
43306         
43307         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43308         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43309         
43310         
43311         
43312         if(this.boxLabel){
43313             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43314         //    viewEl.on('click', this.onClick,  this); 
43315         }
43316          if(this.checked){
43317             this.el.dom.checked =   'checked' ;
43318         }
43319          
43320     } 
43321     
43322     
43323 });//<script type="text/javascript">
43324
43325 /*
43326  * Based  Ext JS Library 1.1.1
43327  * Copyright(c) 2006-2007, Ext JS, LLC.
43328  * LGPL
43329  *
43330  */
43331  
43332 /**
43333  * @class Roo.HtmlEditorCore
43334  * @extends Roo.Component
43335  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43336  *
43337  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43338  */
43339
43340 Roo.HtmlEditorCore = function(config){
43341     
43342     
43343     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43344     
43345     
43346     this.addEvents({
43347         /**
43348          * @event initialize
43349          * Fires when the editor is fully initialized (including the iframe)
43350          * @param {Roo.HtmlEditorCore} this
43351          */
43352         initialize: true,
43353         /**
43354          * @event activate
43355          * Fires when the editor is first receives the focus. Any insertion must wait
43356          * until after this event.
43357          * @param {Roo.HtmlEditorCore} this
43358          */
43359         activate: true,
43360          /**
43361          * @event beforesync
43362          * Fires before the textarea is updated with content from the editor iframe. Return false
43363          * to cancel the sync.
43364          * @param {Roo.HtmlEditorCore} this
43365          * @param {String} html
43366          */
43367         beforesync: true,
43368          /**
43369          * @event beforepush
43370          * Fires before the iframe editor is updated with content from the textarea. Return false
43371          * to cancel the push.
43372          * @param {Roo.HtmlEditorCore} this
43373          * @param {String} html
43374          */
43375         beforepush: true,
43376          /**
43377          * @event sync
43378          * Fires when the textarea is updated with content from the editor iframe.
43379          * @param {Roo.HtmlEditorCore} this
43380          * @param {String} html
43381          */
43382         sync: true,
43383          /**
43384          * @event push
43385          * Fires when the iframe editor is updated with content from the textarea.
43386          * @param {Roo.HtmlEditorCore} this
43387          * @param {String} html
43388          */
43389         push: true,
43390         
43391         /**
43392          * @event editorevent
43393          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43394          * @param {Roo.HtmlEditorCore} this
43395          */
43396         editorevent: true
43397         
43398     });
43399     
43400     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43401     
43402     // defaults : white / black...
43403     this.applyBlacklists();
43404     
43405     
43406     
43407 };
43408
43409
43410 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43411
43412
43413      /**
43414      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43415      */
43416     
43417     owner : false,
43418     
43419      /**
43420      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43421      *                        Roo.resizable.
43422      */
43423     resizable : false,
43424      /**
43425      * @cfg {Number} height (in pixels)
43426      */   
43427     height: 300,
43428    /**
43429      * @cfg {Number} width (in pixels)
43430      */   
43431     width: 500,
43432     
43433     /**
43434      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43435      * 
43436      */
43437     stylesheets: false,
43438     
43439     // id of frame..
43440     frameId: false,
43441     
43442     // private properties
43443     validationEvent : false,
43444     deferHeight: true,
43445     initialized : false,
43446     activated : false,
43447     sourceEditMode : false,
43448     onFocus : Roo.emptyFn,
43449     iframePad:3,
43450     hideMode:'offsets',
43451     
43452     clearUp: true,
43453     
43454     // blacklist + whitelisted elements..
43455     black: false,
43456     white: false,
43457      
43458     bodyCls : '',
43459
43460     /**
43461      * Protected method that will not generally be called directly. It
43462      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43463      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43464      */
43465     getDocMarkup : function(){
43466         // body styles..
43467         var st = '';
43468         
43469         // inherit styels from page...?? 
43470         if (this.stylesheets === false) {
43471             
43472             Roo.get(document.head).select('style').each(function(node) {
43473                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43474             });
43475             
43476             Roo.get(document.head).select('link').each(function(node) { 
43477                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43478             });
43479             
43480         } else if (!this.stylesheets.length) {
43481                 // simple..
43482                 st = '<style type="text/css">' +
43483                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43484                    '</style>';
43485         } else { 
43486             st = '<style type="text/css">' +
43487                     this.stylesheets +
43488                 '</style>';
43489         }
43490         
43491         st +=  '<style type="text/css">' +
43492             'IMG { cursor: pointer } ' +
43493         '</style>';
43494
43495         var cls = 'roo-htmleditor-body';
43496         
43497         if(this.bodyCls.length){
43498             cls += ' ' + this.bodyCls;
43499         }
43500         
43501         return '<html><head>' + st  +
43502             //<style type="text/css">' +
43503             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43504             //'</style>' +
43505             ' </head><body class="' +  cls + '"></body></html>';
43506     },
43507
43508     // private
43509     onRender : function(ct, position)
43510     {
43511         var _t = this;
43512         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43513         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43514         
43515         
43516         this.el.dom.style.border = '0 none';
43517         this.el.dom.setAttribute('tabIndex', -1);
43518         this.el.addClass('x-hidden hide');
43519         
43520         
43521         
43522         if(Roo.isIE){ // fix IE 1px bogus margin
43523             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43524         }
43525        
43526         
43527         this.frameId = Roo.id();
43528         
43529          
43530         
43531         var iframe = this.owner.wrap.createChild({
43532             tag: 'iframe',
43533             cls: 'form-control', // bootstrap..
43534             id: this.frameId,
43535             name: this.frameId,
43536             frameBorder : 'no',
43537             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43538         }, this.el
43539         );
43540         
43541         
43542         this.iframe = iframe.dom;
43543
43544          this.assignDocWin();
43545         
43546         this.doc.designMode = 'on';
43547        
43548         this.doc.open();
43549         this.doc.write(this.getDocMarkup());
43550         this.doc.close();
43551
43552         
43553         var task = { // must defer to wait for browser to be ready
43554             run : function(){
43555                 //console.log("run task?" + this.doc.readyState);
43556                 this.assignDocWin();
43557                 if(this.doc.body || this.doc.readyState == 'complete'){
43558                     try {
43559                         this.doc.designMode="on";
43560                     } catch (e) {
43561                         return;
43562                     }
43563                     Roo.TaskMgr.stop(task);
43564                     this.initEditor.defer(10, this);
43565                 }
43566             },
43567             interval : 10,
43568             duration: 10000,
43569             scope: this
43570         };
43571         Roo.TaskMgr.start(task);
43572
43573     },
43574
43575     // private
43576     onResize : function(w, h)
43577     {
43578          Roo.log('resize: ' +w + ',' + h );
43579         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43580         if(!this.iframe){
43581             return;
43582         }
43583         if(typeof w == 'number'){
43584             
43585             this.iframe.style.width = w + 'px';
43586         }
43587         if(typeof h == 'number'){
43588             
43589             this.iframe.style.height = h + 'px';
43590             if(this.doc){
43591                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43592             }
43593         }
43594         
43595     },
43596
43597     /**
43598      * Toggles the editor between standard and source edit mode.
43599      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43600      */
43601     toggleSourceEdit : function(sourceEditMode){
43602         
43603         this.sourceEditMode = sourceEditMode === true;
43604         
43605         if(this.sourceEditMode){
43606  
43607             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43608             
43609         }else{
43610             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43611             //this.iframe.className = '';
43612             this.deferFocus();
43613         }
43614         //this.setSize(this.owner.wrap.getSize());
43615         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43616     },
43617
43618     
43619   
43620
43621     /**
43622      * Protected method that will not generally be called directly. If you need/want
43623      * custom HTML cleanup, this is the method you should override.
43624      * @param {String} html The HTML to be cleaned
43625      * return {String} The cleaned HTML
43626      */
43627     cleanHtml : function(html){
43628         html = String(html);
43629         if(html.length > 5){
43630             if(Roo.isSafari){ // strip safari nonsense
43631                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43632             }
43633         }
43634         if(html == '&nbsp;'){
43635             html = '';
43636         }
43637         return html;
43638     },
43639
43640     /**
43641      * HTML Editor -> Textarea
43642      * Protected method that will not generally be called directly. Syncs the contents
43643      * of the editor iframe with the textarea.
43644      */
43645     syncValue : function(){
43646         if(this.initialized){
43647             var bd = (this.doc.body || this.doc.documentElement);
43648             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43649             var html = bd.innerHTML;
43650             if(Roo.isSafari){
43651                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43652                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43653                 if(m && m[1]){
43654                     html = '<div style="'+m[0]+'">' + html + '</div>';
43655                 }
43656             }
43657             html = this.cleanHtml(html);
43658             // fix up the special chars.. normaly like back quotes in word...
43659             // however we do not want to do this with chinese..
43660             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43661                 
43662                 var cc = match.charCodeAt();
43663
43664                 // Get the character value, handling surrogate pairs
43665                 if (match.length == 2) {
43666                     // It's a surrogate pair, calculate the Unicode code point
43667                     var high = match.charCodeAt(0) - 0xD800;
43668                     var low  = match.charCodeAt(1) - 0xDC00;
43669                     cc = (high * 0x400) + low + 0x10000;
43670                 }  else if (
43671                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43672                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43673                     (cc >= 0xf900 && cc < 0xfb00 )
43674                 ) {
43675                         return match;
43676                 }  
43677          
43678                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43679                 return "&#" + cc + ";";
43680                 
43681                 
43682             });
43683             
43684             
43685              
43686             if(this.owner.fireEvent('beforesync', this, html) !== false){
43687                 this.el.dom.value = html;
43688                 this.owner.fireEvent('sync', this, html);
43689             }
43690         }
43691     },
43692
43693     /**
43694      * Protected method that will not generally be called directly. Pushes the value of the textarea
43695      * into the iframe editor.
43696      */
43697     pushValue : function(){
43698         if(this.initialized){
43699             var v = this.el.dom.value.trim();
43700             
43701 //            if(v.length < 1){
43702 //                v = '&#160;';
43703 //            }
43704             
43705             if(this.owner.fireEvent('beforepush', this, v) !== false){
43706                 var d = (this.doc.body || this.doc.documentElement);
43707                 d.innerHTML = v;
43708                 this.cleanUpPaste();
43709                 this.el.dom.value = d.innerHTML;
43710                 this.owner.fireEvent('push', this, v);
43711             }
43712         }
43713     },
43714
43715     // private
43716     deferFocus : function(){
43717         this.focus.defer(10, this);
43718     },
43719
43720     // doc'ed in Field
43721     focus : function(){
43722         if(this.win && !this.sourceEditMode){
43723             this.win.focus();
43724         }else{
43725             this.el.focus();
43726         }
43727     },
43728     
43729     assignDocWin: function()
43730     {
43731         var iframe = this.iframe;
43732         
43733          if(Roo.isIE){
43734             this.doc = iframe.contentWindow.document;
43735             this.win = iframe.contentWindow;
43736         } else {
43737 //            if (!Roo.get(this.frameId)) {
43738 //                return;
43739 //            }
43740 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43741 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43742             
43743             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43744                 return;
43745             }
43746             
43747             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43748             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43749         }
43750     },
43751     
43752     // private
43753     initEditor : function(){
43754         //console.log("INIT EDITOR");
43755         this.assignDocWin();
43756         
43757         
43758         
43759         this.doc.designMode="on";
43760         this.doc.open();
43761         this.doc.write(this.getDocMarkup());
43762         this.doc.close();
43763         
43764         var dbody = (this.doc.body || this.doc.documentElement);
43765         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43766         // this copies styles from the containing element into thsi one..
43767         // not sure why we need all of this..
43768         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43769         
43770         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43771         //ss['background-attachment'] = 'fixed'; // w3c
43772         dbody.bgProperties = 'fixed'; // ie
43773         //Roo.DomHelper.applyStyles(dbody, ss);
43774         Roo.EventManager.on(this.doc, {
43775             //'mousedown': this.onEditorEvent,
43776             'mouseup': this.onEditorEvent,
43777             'dblclick': this.onEditorEvent,
43778             'click': this.onEditorEvent,
43779             'keyup': this.onEditorEvent,
43780             buffer:100,
43781             scope: this
43782         });
43783         if(Roo.isGecko){
43784             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43785         }
43786         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43787             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43788         }
43789         this.initialized = true;
43790
43791         this.owner.fireEvent('initialize', this);
43792         this.pushValue();
43793     },
43794
43795     // private
43796     onDestroy : function(){
43797         
43798         
43799         
43800         if(this.rendered){
43801             
43802             //for (var i =0; i < this.toolbars.length;i++) {
43803             //    // fixme - ask toolbars for heights?
43804             //    this.toolbars[i].onDestroy();
43805            // }
43806             
43807             //this.wrap.dom.innerHTML = '';
43808             //this.wrap.remove();
43809         }
43810     },
43811
43812     // private
43813     onFirstFocus : function(){
43814         
43815         this.assignDocWin();
43816         
43817         
43818         this.activated = true;
43819          
43820     
43821         if(Roo.isGecko){ // prevent silly gecko errors
43822             this.win.focus();
43823             var s = this.win.getSelection();
43824             if(!s.focusNode || s.focusNode.nodeType != 3){
43825                 var r = s.getRangeAt(0);
43826                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43827                 r.collapse(true);
43828                 this.deferFocus();
43829             }
43830             try{
43831                 this.execCmd('useCSS', true);
43832                 this.execCmd('styleWithCSS', false);
43833             }catch(e){}
43834         }
43835         this.owner.fireEvent('activate', this);
43836     },
43837
43838     // private
43839     adjustFont: function(btn){
43840         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43841         //if(Roo.isSafari){ // safari
43842         //    adjust *= 2;
43843        // }
43844         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43845         if(Roo.isSafari){ // safari
43846             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43847             v =  (v < 10) ? 10 : v;
43848             v =  (v > 48) ? 48 : v;
43849             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43850             
43851         }
43852         
43853         
43854         v = Math.max(1, v+adjust);
43855         
43856         this.execCmd('FontSize', v  );
43857     },
43858
43859     onEditorEvent : function(e)
43860     {
43861         this.owner.fireEvent('editorevent', this, e);
43862       //  this.updateToolbar();
43863         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43864     },
43865
43866     insertTag : function(tg)
43867     {
43868         // could be a bit smarter... -> wrap the current selected tRoo..
43869         if (tg.toLowerCase() == 'span' ||
43870             tg.toLowerCase() == 'code' ||
43871             tg.toLowerCase() == 'sup' ||
43872             tg.toLowerCase() == 'sub' 
43873             ) {
43874             
43875             range = this.createRange(this.getSelection());
43876             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43877             wrappingNode.appendChild(range.extractContents());
43878             range.insertNode(wrappingNode);
43879
43880             return;
43881             
43882             
43883             
43884         }
43885         this.execCmd("formatblock",   tg);
43886         
43887     },
43888     
43889     insertText : function(txt)
43890     {
43891         
43892         
43893         var range = this.createRange();
43894         range.deleteContents();
43895                //alert(Sender.getAttribute('label'));
43896                
43897         range.insertNode(this.doc.createTextNode(txt));
43898     } ,
43899     
43900      
43901
43902     /**
43903      * Executes a Midas editor command on the editor document and performs necessary focus and
43904      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43905      * @param {String} cmd The Midas command
43906      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43907      */
43908     relayCmd : function(cmd, value){
43909         this.win.focus();
43910         this.execCmd(cmd, value);
43911         this.owner.fireEvent('editorevent', this);
43912         //this.updateToolbar();
43913         this.owner.deferFocus();
43914     },
43915
43916     /**
43917      * Executes a Midas editor command directly on the editor document.
43918      * For visual commands, you should use {@link #relayCmd} instead.
43919      * <b>This should only be called after the editor is initialized.</b>
43920      * @param {String} cmd The Midas command
43921      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43922      */
43923     execCmd : function(cmd, value){
43924         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43925         this.syncValue();
43926     },
43927  
43928  
43929    
43930     /**
43931      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43932      * to insert tRoo.
43933      * @param {String} text | dom node.. 
43934      */
43935     insertAtCursor : function(text)
43936     {
43937         
43938         if(!this.activated){
43939             return;
43940         }
43941         /*
43942         if(Roo.isIE){
43943             this.win.focus();
43944             var r = this.doc.selection.createRange();
43945             if(r){
43946                 r.collapse(true);
43947                 r.pasteHTML(text);
43948                 this.syncValue();
43949                 this.deferFocus();
43950             
43951             }
43952             return;
43953         }
43954         */
43955         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43956             this.win.focus();
43957             
43958             
43959             // from jquery ui (MIT licenced)
43960             var range, node;
43961             var win = this.win;
43962             
43963             if (win.getSelection && win.getSelection().getRangeAt) {
43964                 range = win.getSelection().getRangeAt(0);
43965                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43966                 range.insertNode(node);
43967             } else if (win.document.selection && win.document.selection.createRange) {
43968                 // no firefox support
43969                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43970                 win.document.selection.createRange().pasteHTML(txt);
43971             } else {
43972                 // no firefox support
43973                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43974                 this.execCmd('InsertHTML', txt);
43975             } 
43976             
43977             this.syncValue();
43978             
43979             this.deferFocus();
43980         }
43981     },
43982  // private
43983     mozKeyPress : function(e){
43984         if(e.ctrlKey){
43985             var c = e.getCharCode(), cmd;
43986           
43987             if(c > 0){
43988                 c = String.fromCharCode(c).toLowerCase();
43989                 switch(c){
43990                     case 'b':
43991                         cmd = 'bold';
43992                         break;
43993                     case 'i':
43994                         cmd = 'italic';
43995                         break;
43996                     
43997                     case 'u':
43998                         cmd = 'underline';
43999                         break;
44000                     
44001                     case 'v':
44002                         this.cleanUpPaste.defer(100, this);
44003                         return;
44004                         
44005                 }
44006                 if(cmd){
44007                     this.win.focus();
44008                     this.execCmd(cmd);
44009                     this.deferFocus();
44010                     e.preventDefault();
44011                 }
44012                 
44013             }
44014         }
44015     },
44016
44017     // private
44018     fixKeys : function(){ // load time branching for fastest keydown performance
44019         if(Roo.isIE){
44020             return function(e){
44021                 var k = e.getKey(), r;
44022                 if(k == e.TAB){
44023                     e.stopEvent();
44024                     r = this.doc.selection.createRange();
44025                     if(r){
44026                         r.collapse(true);
44027                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44028                         this.deferFocus();
44029                     }
44030                     return;
44031                 }
44032                 
44033                 if(k == e.ENTER){
44034                     r = this.doc.selection.createRange();
44035                     if(r){
44036                         var target = r.parentElement();
44037                         if(!target || target.tagName.toLowerCase() != 'li'){
44038                             e.stopEvent();
44039                             r.pasteHTML('<br />');
44040                             r.collapse(false);
44041                             r.select();
44042                         }
44043                     }
44044                 }
44045                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44046                     this.cleanUpPaste.defer(100, this);
44047                     return;
44048                 }
44049                 
44050                 
44051             };
44052         }else if(Roo.isOpera){
44053             return function(e){
44054                 var k = e.getKey();
44055                 if(k == e.TAB){
44056                     e.stopEvent();
44057                     this.win.focus();
44058                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44059                     this.deferFocus();
44060                 }
44061                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44062                     this.cleanUpPaste.defer(100, this);
44063                     return;
44064                 }
44065                 
44066             };
44067         }else if(Roo.isSafari){
44068             return function(e){
44069                 var k = e.getKey();
44070                 
44071                 if(k == e.TAB){
44072                     e.stopEvent();
44073                     this.execCmd('InsertText','\t');
44074                     this.deferFocus();
44075                     return;
44076                 }
44077                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44078                     this.cleanUpPaste.defer(100, this);
44079                     return;
44080                 }
44081                 
44082              };
44083         }
44084     }(),
44085     
44086     getAllAncestors: function()
44087     {
44088         var p = this.getSelectedNode();
44089         var a = [];
44090         if (!p) {
44091             a.push(p); // push blank onto stack..
44092             p = this.getParentElement();
44093         }
44094         
44095         
44096         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44097             a.push(p);
44098             p = p.parentNode;
44099         }
44100         a.push(this.doc.body);
44101         return a;
44102     },
44103     lastSel : false,
44104     lastSelNode : false,
44105     
44106     
44107     getSelection : function() 
44108     {
44109         this.assignDocWin();
44110         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44111     },
44112     
44113     getSelectedNode: function() 
44114     {
44115         // this may only work on Gecko!!!
44116         
44117         // should we cache this!!!!
44118         
44119         
44120         
44121          
44122         var range = this.createRange(this.getSelection()).cloneRange();
44123         
44124         if (Roo.isIE) {
44125             var parent = range.parentElement();
44126             while (true) {
44127                 var testRange = range.duplicate();
44128                 testRange.moveToElementText(parent);
44129                 if (testRange.inRange(range)) {
44130                     break;
44131                 }
44132                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44133                     break;
44134                 }
44135                 parent = parent.parentElement;
44136             }
44137             return parent;
44138         }
44139         
44140         // is ancestor a text element.
44141         var ac =  range.commonAncestorContainer;
44142         if (ac.nodeType == 3) {
44143             ac = ac.parentNode;
44144         }
44145         
44146         var ar = ac.childNodes;
44147          
44148         var nodes = [];
44149         var other_nodes = [];
44150         var has_other_nodes = false;
44151         for (var i=0;i<ar.length;i++) {
44152             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44153                 continue;
44154             }
44155             // fullly contained node.
44156             
44157             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44158                 nodes.push(ar[i]);
44159                 continue;
44160             }
44161             
44162             // probably selected..
44163             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44164                 other_nodes.push(ar[i]);
44165                 continue;
44166             }
44167             // outer..
44168             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44169                 continue;
44170             }
44171             
44172             
44173             has_other_nodes = true;
44174         }
44175         if (!nodes.length && other_nodes.length) {
44176             nodes= other_nodes;
44177         }
44178         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44179             return false;
44180         }
44181         
44182         return nodes[0];
44183     },
44184     createRange: function(sel)
44185     {
44186         // this has strange effects when using with 
44187         // top toolbar - not sure if it's a great idea.
44188         //this.editor.contentWindow.focus();
44189         if (typeof sel != "undefined") {
44190             try {
44191                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44192             } catch(e) {
44193                 return this.doc.createRange();
44194             }
44195         } else {
44196             return this.doc.createRange();
44197         }
44198     },
44199     getParentElement: function()
44200     {
44201         
44202         this.assignDocWin();
44203         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44204         
44205         var range = this.createRange(sel);
44206          
44207         try {
44208             var p = range.commonAncestorContainer;
44209             while (p.nodeType == 3) { // text node
44210                 p = p.parentNode;
44211             }
44212             return p;
44213         } catch (e) {
44214             return null;
44215         }
44216     
44217     },
44218     /***
44219      *
44220      * Range intersection.. the hard stuff...
44221      *  '-1' = before
44222      *  '0' = hits..
44223      *  '1' = after.
44224      *         [ -- selected range --- ]
44225      *   [fail]                        [fail]
44226      *
44227      *    basically..
44228      *      if end is before start or  hits it. fail.
44229      *      if start is after end or hits it fail.
44230      *
44231      *   if either hits (but other is outside. - then it's not 
44232      *   
44233      *    
44234      **/
44235     
44236     
44237     // @see http://www.thismuchiknow.co.uk/?p=64.
44238     rangeIntersectsNode : function(range, node)
44239     {
44240         var nodeRange = node.ownerDocument.createRange();
44241         try {
44242             nodeRange.selectNode(node);
44243         } catch (e) {
44244             nodeRange.selectNodeContents(node);
44245         }
44246     
44247         var rangeStartRange = range.cloneRange();
44248         rangeStartRange.collapse(true);
44249     
44250         var rangeEndRange = range.cloneRange();
44251         rangeEndRange.collapse(false);
44252     
44253         var nodeStartRange = nodeRange.cloneRange();
44254         nodeStartRange.collapse(true);
44255     
44256         var nodeEndRange = nodeRange.cloneRange();
44257         nodeEndRange.collapse(false);
44258     
44259         return rangeStartRange.compareBoundaryPoints(
44260                  Range.START_TO_START, nodeEndRange) == -1 &&
44261                rangeEndRange.compareBoundaryPoints(
44262                  Range.START_TO_START, nodeStartRange) == 1;
44263         
44264          
44265     },
44266     rangeCompareNode : function(range, node)
44267     {
44268         var nodeRange = node.ownerDocument.createRange();
44269         try {
44270             nodeRange.selectNode(node);
44271         } catch (e) {
44272             nodeRange.selectNodeContents(node);
44273         }
44274         
44275         
44276         range.collapse(true);
44277     
44278         nodeRange.collapse(true);
44279      
44280         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44281         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44282          
44283         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44284         
44285         var nodeIsBefore   =  ss == 1;
44286         var nodeIsAfter    = ee == -1;
44287         
44288         if (nodeIsBefore && nodeIsAfter) {
44289             return 0; // outer
44290         }
44291         if (!nodeIsBefore && nodeIsAfter) {
44292             return 1; //right trailed.
44293         }
44294         
44295         if (nodeIsBefore && !nodeIsAfter) {
44296             return 2;  // left trailed.
44297         }
44298         // fully contined.
44299         return 3;
44300     },
44301
44302     // private? - in a new class?
44303     cleanUpPaste :  function()
44304     {
44305         // cleans up the whole document..
44306         Roo.log('cleanuppaste');
44307         
44308         this.cleanUpChildren(this.doc.body);
44309         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44310         if (clean != this.doc.body.innerHTML) {
44311             this.doc.body.innerHTML = clean;
44312         }
44313         
44314     },
44315     
44316     cleanWordChars : function(input) {// change the chars to hex code
44317         var he = Roo.HtmlEditorCore;
44318         
44319         var output = input;
44320         Roo.each(he.swapCodes, function(sw) { 
44321             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44322             
44323             output = output.replace(swapper, sw[1]);
44324         });
44325         
44326         return output;
44327     },
44328     
44329     
44330     cleanUpChildren : function (n)
44331     {
44332         if (!n.childNodes.length) {
44333             return;
44334         }
44335         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44336            this.cleanUpChild(n.childNodes[i]);
44337         }
44338     },
44339     
44340     
44341         
44342     
44343     cleanUpChild : function (node)
44344     {
44345         var ed = this;
44346         //console.log(node);
44347         if (node.nodeName == "#text") {
44348             // clean up silly Windows -- stuff?
44349             return; 
44350         }
44351         if (node.nodeName == "#comment") {
44352             node.parentNode.removeChild(node);
44353             // clean up silly Windows -- stuff?
44354             return; 
44355         }
44356         var lcname = node.tagName.toLowerCase();
44357         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44358         // whitelist of tags..
44359         
44360         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44361             // remove node.
44362             node.parentNode.removeChild(node);
44363             return;
44364             
44365         }
44366         
44367         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44368         
44369         // spans with no attributes - just remove them..
44370         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44371             remove_keep_children = true;
44372         }
44373         
44374         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44375         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44376         
44377         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44378         //    remove_keep_children = true;
44379         //}
44380         
44381         if (remove_keep_children) {
44382             this.cleanUpChildren(node);
44383             // inserts everything just before this node...
44384             while (node.childNodes.length) {
44385                 var cn = node.childNodes[0];
44386                 node.removeChild(cn);
44387                 node.parentNode.insertBefore(cn, node);
44388             }
44389             node.parentNode.removeChild(node);
44390             return;
44391         }
44392         
44393         if (!node.attributes || !node.attributes.length) {
44394             
44395           
44396             
44397             
44398             this.cleanUpChildren(node);
44399             return;
44400         }
44401         
44402         function cleanAttr(n,v)
44403         {
44404             
44405             if (v.match(/^\./) || v.match(/^\//)) {
44406                 return;
44407             }
44408             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44409                 return;
44410             }
44411             if (v.match(/^#/)) {
44412                 return;
44413             }
44414 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44415             node.removeAttribute(n);
44416             
44417         }
44418         
44419         var cwhite = this.cwhite;
44420         var cblack = this.cblack;
44421             
44422         function cleanStyle(n,v)
44423         {
44424             if (v.match(/expression/)) { //XSS?? should we even bother..
44425                 node.removeAttribute(n);
44426                 return;
44427             }
44428             
44429             var parts = v.split(/;/);
44430             var clean = [];
44431             
44432             Roo.each(parts, function(p) {
44433                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44434                 if (!p.length) {
44435                     return true;
44436                 }
44437                 var l = p.split(':').shift().replace(/\s+/g,'');
44438                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44439                 
44440                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44441 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44442                     //node.removeAttribute(n);
44443                     return true;
44444                 }
44445                 //Roo.log()
44446                 // only allow 'c whitelisted system attributes'
44447                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44448 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44449                     //node.removeAttribute(n);
44450                     return true;
44451                 }
44452                 
44453                 
44454                  
44455                 
44456                 clean.push(p);
44457                 return true;
44458             });
44459             if (clean.length) { 
44460                 node.setAttribute(n, clean.join(';'));
44461             } else {
44462                 node.removeAttribute(n);
44463             }
44464             
44465         }
44466         
44467         
44468         for (var i = node.attributes.length-1; i > -1 ; i--) {
44469             var a = node.attributes[i];
44470             //console.log(a);
44471             
44472             if (a.name.toLowerCase().substr(0,2)=='on')  {
44473                 node.removeAttribute(a.name);
44474                 continue;
44475             }
44476             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44477                 node.removeAttribute(a.name);
44478                 continue;
44479             }
44480             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44481                 cleanAttr(a.name,a.value); // fixme..
44482                 continue;
44483             }
44484             if (a.name == 'style') {
44485                 cleanStyle(a.name,a.value);
44486                 continue;
44487             }
44488             /// clean up MS crap..
44489             // tecnically this should be a list of valid class'es..
44490             
44491             
44492             if (a.name == 'class') {
44493                 if (a.value.match(/^Mso/)) {
44494                     node.removeAttribute('class');
44495                 }
44496                 
44497                 if (a.value.match(/^body$/)) {
44498                     node.removeAttribute('class');
44499                 }
44500                 continue;
44501             }
44502             
44503             // style cleanup!?
44504             // class cleanup?
44505             
44506         }
44507         
44508         
44509         this.cleanUpChildren(node);
44510         
44511         
44512     },
44513     
44514     /**
44515      * Clean up MS wordisms...
44516      */
44517     cleanWord : function(node)
44518     {
44519         if (!node) {
44520             this.cleanWord(this.doc.body);
44521             return;
44522         }
44523         
44524         if(
44525                 node.nodeName == 'SPAN' &&
44526                 !node.hasAttributes() &&
44527                 node.childNodes.length == 1 &&
44528                 node.firstChild.nodeName == "#text"  
44529         ) {
44530             var textNode = node.firstChild;
44531             node.removeChild(textNode);
44532             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44533                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44534             }
44535             node.parentNode.insertBefore(textNode, node);
44536             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44537                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44538             }
44539             node.parentNode.removeChild(node);
44540         }
44541         
44542         if (node.nodeName == "#text") {
44543             // clean up silly Windows -- stuff?
44544             return; 
44545         }
44546         if (node.nodeName == "#comment") {
44547             node.parentNode.removeChild(node);
44548             // clean up silly Windows -- stuff?
44549             return; 
44550         }
44551         
44552         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44553             node.parentNode.removeChild(node);
44554             return;
44555         }
44556         //Roo.log(node.tagName);
44557         // remove - but keep children..
44558         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44559             //Roo.log('-- removed');
44560             while (node.childNodes.length) {
44561                 var cn = node.childNodes[0];
44562                 node.removeChild(cn);
44563                 node.parentNode.insertBefore(cn, node);
44564                 // move node to parent - and clean it..
44565                 this.cleanWord(cn);
44566             }
44567             node.parentNode.removeChild(node);
44568             /// no need to iterate chidlren = it's got none..
44569             //this.iterateChildren(node, this.cleanWord);
44570             return;
44571         }
44572         // clean styles
44573         if (node.className.length) {
44574             
44575             var cn = node.className.split(/\W+/);
44576             var cna = [];
44577             Roo.each(cn, function(cls) {
44578                 if (cls.match(/Mso[a-zA-Z]+/)) {
44579                     return;
44580                 }
44581                 cna.push(cls);
44582             });
44583             node.className = cna.length ? cna.join(' ') : '';
44584             if (!cna.length) {
44585                 node.removeAttribute("class");
44586             }
44587         }
44588         
44589         if (node.hasAttribute("lang")) {
44590             node.removeAttribute("lang");
44591         }
44592         
44593         if (node.hasAttribute("style")) {
44594             
44595             var styles = node.getAttribute("style").split(";");
44596             var nstyle = [];
44597             Roo.each(styles, function(s) {
44598                 if (!s.match(/:/)) {
44599                     return;
44600                 }
44601                 var kv = s.split(":");
44602                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44603                     return;
44604                 }
44605                 // what ever is left... we allow.
44606                 nstyle.push(s);
44607             });
44608             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44609             if (!nstyle.length) {
44610                 node.removeAttribute('style');
44611             }
44612         }
44613         this.iterateChildren(node, this.cleanWord);
44614         
44615         
44616         
44617     },
44618     /**
44619      * iterateChildren of a Node, calling fn each time, using this as the scole..
44620      * @param {DomNode} node node to iterate children of.
44621      * @param {Function} fn method of this class to call on each item.
44622      */
44623     iterateChildren : function(node, fn)
44624     {
44625         if (!node.childNodes.length) {
44626                 return;
44627         }
44628         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44629            fn.call(this, node.childNodes[i])
44630         }
44631     },
44632     
44633     
44634     /**
44635      * cleanTableWidths.
44636      *
44637      * Quite often pasting from word etc.. results in tables with column and widths.
44638      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44639      *
44640      */
44641     cleanTableWidths : function(node)
44642     {
44643          
44644          
44645         if (!node) {
44646             this.cleanTableWidths(this.doc.body);
44647             return;
44648         }
44649         
44650         // ignore list...
44651         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44652             return; 
44653         }
44654         Roo.log(node.tagName);
44655         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44656             this.iterateChildren(node, this.cleanTableWidths);
44657             return;
44658         }
44659         if (node.hasAttribute('width')) {
44660             node.removeAttribute('width');
44661         }
44662         
44663          
44664         if (node.hasAttribute("style")) {
44665             // pretty basic...
44666             
44667             var styles = node.getAttribute("style").split(";");
44668             var nstyle = [];
44669             Roo.each(styles, function(s) {
44670                 if (!s.match(/:/)) {
44671                     return;
44672                 }
44673                 var kv = s.split(":");
44674                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44675                     return;
44676                 }
44677                 // what ever is left... we allow.
44678                 nstyle.push(s);
44679             });
44680             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44681             if (!nstyle.length) {
44682                 node.removeAttribute('style');
44683             }
44684         }
44685         
44686         this.iterateChildren(node, this.cleanTableWidths);
44687         
44688         
44689     },
44690     
44691     
44692     
44693     
44694     domToHTML : function(currentElement, depth, nopadtext) {
44695         
44696         depth = depth || 0;
44697         nopadtext = nopadtext || false;
44698     
44699         if (!currentElement) {
44700             return this.domToHTML(this.doc.body);
44701         }
44702         
44703         //Roo.log(currentElement);
44704         var j;
44705         var allText = false;
44706         var nodeName = currentElement.nodeName;
44707         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44708         
44709         if  (nodeName == '#text') {
44710             
44711             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44712         }
44713         
44714         
44715         var ret = '';
44716         if (nodeName != 'BODY') {
44717              
44718             var i = 0;
44719             // Prints the node tagName, such as <A>, <IMG>, etc
44720             if (tagName) {
44721                 var attr = [];
44722                 for(i = 0; i < currentElement.attributes.length;i++) {
44723                     // quoting?
44724                     var aname = currentElement.attributes.item(i).name;
44725                     if (!currentElement.attributes.item(i).value.length) {
44726                         continue;
44727                     }
44728                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44729                 }
44730                 
44731                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44732             } 
44733             else {
44734                 
44735                 // eack
44736             }
44737         } else {
44738             tagName = false;
44739         }
44740         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44741             return ret;
44742         }
44743         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44744             nopadtext = true;
44745         }
44746         
44747         
44748         // Traverse the tree
44749         i = 0;
44750         var currentElementChild = currentElement.childNodes.item(i);
44751         var allText = true;
44752         var innerHTML  = '';
44753         lastnode = '';
44754         while (currentElementChild) {
44755             // Formatting code (indent the tree so it looks nice on the screen)
44756             var nopad = nopadtext;
44757             if (lastnode == 'SPAN') {
44758                 nopad  = true;
44759             }
44760             // text
44761             if  (currentElementChild.nodeName == '#text') {
44762                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44763                 toadd = nopadtext ? toadd : toadd.trim();
44764                 if (!nopad && toadd.length > 80) {
44765                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44766                 }
44767                 innerHTML  += toadd;
44768                 
44769                 i++;
44770                 currentElementChild = currentElement.childNodes.item(i);
44771                 lastNode = '';
44772                 continue;
44773             }
44774             allText = false;
44775             
44776             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44777                 
44778             // Recursively traverse the tree structure of the child node
44779             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44780             lastnode = currentElementChild.nodeName;
44781             i++;
44782             currentElementChild=currentElement.childNodes.item(i);
44783         }
44784         
44785         ret += innerHTML;
44786         
44787         if (!allText) {
44788                 // The remaining code is mostly for formatting the tree
44789             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44790         }
44791         
44792         
44793         if (tagName) {
44794             ret+= "</"+tagName+">";
44795         }
44796         return ret;
44797         
44798     },
44799         
44800     applyBlacklists : function()
44801     {
44802         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44803         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44804         
44805         this.white = [];
44806         this.black = [];
44807         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44808             if (b.indexOf(tag) > -1) {
44809                 return;
44810             }
44811             this.white.push(tag);
44812             
44813         }, this);
44814         
44815         Roo.each(w, function(tag) {
44816             if (b.indexOf(tag) > -1) {
44817                 return;
44818             }
44819             if (this.white.indexOf(tag) > -1) {
44820                 return;
44821             }
44822             this.white.push(tag);
44823             
44824         }, this);
44825         
44826         
44827         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44828             if (w.indexOf(tag) > -1) {
44829                 return;
44830             }
44831             this.black.push(tag);
44832             
44833         }, this);
44834         
44835         Roo.each(b, function(tag) {
44836             if (w.indexOf(tag) > -1) {
44837                 return;
44838             }
44839             if (this.black.indexOf(tag) > -1) {
44840                 return;
44841             }
44842             this.black.push(tag);
44843             
44844         }, this);
44845         
44846         
44847         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44848         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44849         
44850         this.cwhite = [];
44851         this.cblack = [];
44852         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44853             if (b.indexOf(tag) > -1) {
44854                 return;
44855             }
44856             this.cwhite.push(tag);
44857             
44858         }, this);
44859         
44860         Roo.each(w, function(tag) {
44861             if (b.indexOf(tag) > -1) {
44862                 return;
44863             }
44864             if (this.cwhite.indexOf(tag) > -1) {
44865                 return;
44866             }
44867             this.cwhite.push(tag);
44868             
44869         }, this);
44870         
44871         
44872         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44873             if (w.indexOf(tag) > -1) {
44874                 return;
44875             }
44876             this.cblack.push(tag);
44877             
44878         }, this);
44879         
44880         Roo.each(b, function(tag) {
44881             if (w.indexOf(tag) > -1) {
44882                 return;
44883             }
44884             if (this.cblack.indexOf(tag) > -1) {
44885                 return;
44886             }
44887             this.cblack.push(tag);
44888             
44889         }, this);
44890     },
44891     
44892     setStylesheets : function(stylesheets)
44893     {
44894         if(typeof(stylesheets) == 'string'){
44895             Roo.get(this.iframe.contentDocument.head).createChild({
44896                 tag : 'link',
44897                 rel : 'stylesheet',
44898                 type : 'text/css',
44899                 href : stylesheets
44900             });
44901             
44902             return;
44903         }
44904         var _this = this;
44905      
44906         Roo.each(stylesheets, function(s) {
44907             if(!s.length){
44908                 return;
44909             }
44910             
44911             Roo.get(_this.iframe.contentDocument.head).createChild({
44912                 tag : 'link',
44913                 rel : 'stylesheet',
44914                 type : 'text/css',
44915                 href : s
44916             });
44917         });
44918
44919         
44920     },
44921     
44922     removeStylesheets : function()
44923     {
44924         var _this = this;
44925         
44926         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44927             s.remove();
44928         });
44929     },
44930     
44931     setStyle : function(style)
44932     {
44933         Roo.get(this.iframe.contentDocument.head).createChild({
44934             tag : 'style',
44935             type : 'text/css',
44936             html : style
44937         });
44938
44939         return;
44940     }
44941     
44942     // hide stuff that is not compatible
44943     /**
44944      * @event blur
44945      * @hide
44946      */
44947     /**
44948      * @event change
44949      * @hide
44950      */
44951     /**
44952      * @event focus
44953      * @hide
44954      */
44955     /**
44956      * @event specialkey
44957      * @hide
44958      */
44959     /**
44960      * @cfg {String} fieldClass @hide
44961      */
44962     /**
44963      * @cfg {String} focusClass @hide
44964      */
44965     /**
44966      * @cfg {String} autoCreate @hide
44967      */
44968     /**
44969      * @cfg {String} inputType @hide
44970      */
44971     /**
44972      * @cfg {String} invalidClass @hide
44973      */
44974     /**
44975      * @cfg {String} invalidText @hide
44976      */
44977     /**
44978      * @cfg {String} msgFx @hide
44979      */
44980     /**
44981      * @cfg {String} validateOnBlur @hide
44982      */
44983 });
44984
44985 Roo.HtmlEditorCore.white = [
44986         'area', 'br', 'img', 'input', 'hr', 'wbr',
44987         
44988        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44989        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44990        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44991        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44992        'table',   'ul',         'xmp', 
44993        
44994        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44995       'thead',   'tr', 
44996      
44997       'dir', 'menu', 'ol', 'ul', 'dl',
44998        
44999       'embed',  'object'
45000 ];
45001
45002
45003 Roo.HtmlEditorCore.black = [
45004     //    'embed',  'object', // enable - backend responsiblity to clean thiese
45005         'applet', // 
45006         'base',   'basefont', 'bgsound', 'blink',  'body', 
45007         'frame',  'frameset', 'head',    'html',   'ilayer', 
45008         'iframe', 'layer',  'link',     'meta',    'object',   
45009         'script', 'style' ,'title',  'xml' // clean later..
45010 ];
45011 Roo.HtmlEditorCore.clean = [
45012     'script', 'style', 'title', 'xml'
45013 ];
45014 Roo.HtmlEditorCore.remove = [
45015     'font'
45016 ];
45017 // attributes..
45018
45019 Roo.HtmlEditorCore.ablack = [
45020     'on'
45021 ];
45022     
45023 Roo.HtmlEditorCore.aclean = [ 
45024     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45025 ];
45026
45027 // protocols..
45028 Roo.HtmlEditorCore.pwhite= [
45029         'http',  'https',  'mailto'
45030 ];
45031
45032 // white listed style attributes.
45033 Roo.HtmlEditorCore.cwhite= [
45034       //  'text-align', /// default is to allow most things..
45035       
45036          
45037 //        'font-size'//??
45038 ];
45039
45040 // black listed style attributes.
45041 Roo.HtmlEditorCore.cblack= [
45042       //  'font-size' -- this can be set by the project 
45043 ];
45044
45045
45046 Roo.HtmlEditorCore.swapCodes   =[ 
45047     [    8211, "--" ], 
45048     [    8212, "--" ], 
45049     [    8216,  "'" ],  
45050     [    8217, "'" ],  
45051     [    8220, '"' ],  
45052     [    8221, '"' ],  
45053     [    8226, "*" ],  
45054     [    8230, "..." ]
45055 ]; 
45056
45057     //<script type="text/javascript">
45058
45059 /*
45060  * Ext JS Library 1.1.1
45061  * Copyright(c) 2006-2007, Ext JS, LLC.
45062  * Licence LGPL
45063  * 
45064  */
45065  
45066  
45067 Roo.form.HtmlEditor = function(config){
45068     
45069     
45070     
45071     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45072     
45073     if (!this.toolbars) {
45074         this.toolbars = [];
45075     }
45076     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45077     
45078     
45079 };
45080
45081 /**
45082  * @class Roo.form.HtmlEditor
45083  * @extends Roo.form.Field
45084  * Provides a lightweight HTML Editor component.
45085  *
45086  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45087  * 
45088  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45089  * supported by this editor.</b><br/><br/>
45090  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45091  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45092  */
45093 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45094     /**
45095      * @cfg {Boolean} clearUp
45096      */
45097     clearUp : true,
45098       /**
45099      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45100      */
45101     toolbars : false,
45102    
45103      /**
45104      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45105      *                        Roo.resizable.
45106      */
45107     resizable : false,
45108      /**
45109      * @cfg {Number} height (in pixels)
45110      */   
45111     height: 300,
45112    /**
45113      * @cfg {Number} width (in pixels)
45114      */   
45115     width: 500,
45116     
45117     /**
45118      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45119      * 
45120      */
45121     stylesheets: false,
45122     
45123     
45124      /**
45125      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45126      * 
45127      */
45128     cblack: false,
45129     /**
45130      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45131      * 
45132      */
45133     cwhite: false,
45134     
45135      /**
45136      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45137      * 
45138      */
45139     black: false,
45140     /**
45141      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45142      * 
45143      */
45144     white: false,
45145     
45146     // id of frame..
45147     frameId: false,
45148     
45149     // private properties
45150     validationEvent : false,
45151     deferHeight: true,
45152     initialized : false,
45153     activated : false,
45154     
45155     onFocus : Roo.emptyFn,
45156     iframePad:3,
45157     hideMode:'offsets',
45158     
45159     actionMode : 'container', // defaults to hiding it...
45160     
45161     defaultAutoCreate : { // modified by initCompnoent..
45162         tag: "textarea",
45163         style:"width:500px;height:300px;",
45164         autocomplete: "new-password"
45165     },
45166
45167     // private
45168     initComponent : function(){
45169         this.addEvents({
45170             /**
45171              * @event initialize
45172              * Fires when the editor is fully initialized (including the iframe)
45173              * @param {HtmlEditor} this
45174              */
45175             initialize: true,
45176             /**
45177              * @event activate
45178              * Fires when the editor is first receives the focus. Any insertion must wait
45179              * until after this event.
45180              * @param {HtmlEditor} this
45181              */
45182             activate: true,
45183              /**
45184              * @event beforesync
45185              * Fires before the textarea is updated with content from the editor iframe. Return false
45186              * to cancel the sync.
45187              * @param {HtmlEditor} this
45188              * @param {String} html
45189              */
45190             beforesync: true,
45191              /**
45192              * @event beforepush
45193              * Fires before the iframe editor is updated with content from the textarea. Return false
45194              * to cancel the push.
45195              * @param {HtmlEditor} this
45196              * @param {String} html
45197              */
45198             beforepush: true,
45199              /**
45200              * @event sync
45201              * Fires when the textarea is updated with content from the editor iframe.
45202              * @param {HtmlEditor} this
45203              * @param {String} html
45204              */
45205             sync: true,
45206              /**
45207              * @event push
45208              * Fires when the iframe editor is updated with content from the textarea.
45209              * @param {HtmlEditor} this
45210              * @param {String} html
45211              */
45212             push: true,
45213              /**
45214              * @event editmodechange
45215              * Fires when the editor switches edit modes
45216              * @param {HtmlEditor} this
45217              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45218              */
45219             editmodechange: true,
45220             /**
45221              * @event editorevent
45222              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45223              * @param {HtmlEditor} this
45224              */
45225             editorevent: true,
45226             /**
45227              * @event firstfocus
45228              * Fires when on first focus - needed by toolbars..
45229              * @param {HtmlEditor} this
45230              */
45231             firstfocus: true,
45232             /**
45233              * @event autosave
45234              * Auto save the htmlEditor value as a file into Events
45235              * @param {HtmlEditor} this
45236              */
45237             autosave: true,
45238             /**
45239              * @event savedpreview
45240              * preview the saved version of htmlEditor
45241              * @param {HtmlEditor} this
45242              */
45243             savedpreview: true,
45244             
45245             /**
45246             * @event stylesheetsclick
45247             * Fires when press the Sytlesheets button
45248             * @param {Roo.HtmlEditorCore} this
45249             */
45250             stylesheetsclick: true
45251         });
45252         this.defaultAutoCreate =  {
45253             tag: "textarea",
45254             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45255             autocomplete: "new-password"
45256         };
45257     },
45258
45259     /**
45260      * Protected method that will not generally be called directly. It
45261      * is called when the editor creates its toolbar. Override this method if you need to
45262      * add custom toolbar buttons.
45263      * @param {HtmlEditor} editor
45264      */
45265     createToolbar : function(editor){
45266         Roo.log("create toolbars");
45267         if (!editor.toolbars || !editor.toolbars.length) {
45268             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45269         }
45270         
45271         for (var i =0 ; i < editor.toolbars.length;i++) {
45272             editor.toolbars[i] = Roo.factory(
45273                     typeof(editor.toolbars[i]) == 'string' ?
45274                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45275                 Roo.form.HtmlEditor);
45276             editor.toolbars[i].init(editor);
45277         }
45278          
45279         
45280     },
45281
45282      
45283     // private
45284     onRender : function(ct, position)
45285     {
45286         var _t = this;
45287         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45288         
45289         this.wrap = this.el.wrap({
45290             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45291         });
45292         
45293         this.editorcore.onRender(ct, position);
45294          
45295         if (this.resizable) {
45296             this.resizeEl = new Roo.Resizable(this.wrap, {
45297                 pinned : true,
45298                 wrap: true,
45299                 dynamic : true,
45300                 minHeight : this.height,
45301                 height: this.height,
45302                 handles : this.resizable,
45303                 width: this.width,
45304                 listeners : {
45305                     resize : function(r, w, h) {
45306                         _t.onResize(w,h); // -something
45307                     }
45308                 }
45309             });
45310             
45311         }
45312         this.createToolbar(this);
45313        
45314         
45315         if(!this.width){
45316             this.setSize(this.wrap.getSize());
45317         }
45318         if (this.resizeEl) {
45319             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45320             // should trigger onReize..
45321         }
45322         
45323         this.keyNav = new Roo.KeyNav(this.el, {
45324             
45325             "tab" : function(e){
45326                 e.preventDefault();
45327                 
45328                 var value = this.getValue();
45329                 
45330                 var start = this.el.dom.selectionStart;
45331                 var end = this.el.dom.selectionEnd;
45332                 
45333                 if(!e.shiftKey){
45334                     
45335                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45336                     this.el.dom.setSelectionRange(end + 1, end + 1);
45337                     return;
45338                 }
45339                 
45340                 var f = value.substring(0, start).split("\t");
45341                 
45342                 if(f.pop().length != 0){
45343                     return;
45344                 }
45345                 
45346                 this.setValue(f.join("\t") + value.substring(end));
45347                 this.el.dom.setSelectionRange(start - 1, start - 1);
45348                 
45349             },
45350             
45351             "home" : function(e){
45352                 e.preventDefault();
45353                 
45354                 var curr = this.el.dom.selectionStart;
45355                 var lines = this.getValue().split("\n");
45356                 
45357                 if(!lines.length){
45358                     return;
45359                 }
45360                 
45361                 if(e.ctrlKey){
45362                     this.el.dom.setSelectionRange(0, 0);
45363                     return;
45364                 }
45365                 
45366                 var pos = 0;
45367                 
45368                 for (var i = 0; i < lines.length;i++) {
45369                     pos += lines[i].length;
45370                     
45371                     if(i != 0){
45372                         pos += 1;
45373                     }
45374                     
45375                     if(pos < curr){
45376                         continue;
45377                     }
45378                     
45379                     pos -= lines[i].length;
45380                     
45381                     break;
45382                 }
45383                 
45384                 if(!e.shiftKey){
45385                     this.el.dom.setSelectionRange(pos, pos);
45386                     return;
45387                 }
45388                 
45389                 this.el.dom.selectionStart = pos;
45390                 this.el.dom.selectionEnd = curr;
45391             },
45392             
45393             "end" : function(e){
45394                 e.preventDefault();
45395                 
45396                 var curr = this.el.dom.selectionStart;
45397                 var lines = this.getValue().split("\n");
45398                 
45399                 if(!lines.length){
45400                     return;
45401                 }
45402                 
45403                 if(e.ctrlKey){
45404                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45405                     return;
45406                 }
45407                 
45408                 var pos = 0;
45409                 
45410                 for (var i = 0; i < lines.length;i++) {
45411                     
45412                     pos += lines[i].length;
45413                     
45414                     if(i != 0){
45415                         pos += 1;
45416                     }
45417                     
45418                     if(pos < curr){
45419                         continue;
45420                     }
45421                     
45422                     break;
45423                 }
45424                 
45425                 if(!e.shiftKey){
45426                     this.el.dom.setSelectionRange(pos, pos);
45427                     return;
45428                 }
45429                 
45430                 this.el.dom.selectionStart = curr;
45431                 this.el.dom.selectionEnd = pos;
45432             },
45433
45434             scope : this,
45435
45436             doRelay : function(foo, bar, hname){
45437                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45438             },
45439
45440             forceKeyDown: true
45441         });
45442         
45443 //        if(this.autosave && this.w){
45444 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45445 //        }
45446     },
45447
45448     // private
45449     onResize : function(w, h)
45450     {
45451         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45452         var ew = false;
45453         var eh = false;
45454         
45455         if(this.el ){
45456             if(typeof w == 'number'){
45457                 var aw = w - this.wrap.getFrameWidth('lr');
45458                 this.el.setWidth(this.adjustWidth('textarea', aw));
45459                 ew = aw;
45460             }
45461             if(typeof h == 'number'){
45462                 var tbh = 0;
45463                 for (var i =0; i < this.toolbars.length;i++) {
45464                     // fixme - ask toolbars for heights?
45465                     tbh += this.toolbars[i].tb.el.getHeight();
45466                     if (this.toolbars[i].footer) {
45467                         tbh += this.toolbars[i].footer.el.getHeight();
45468                     }
45469                 }
45470                 
45471                 
45472                 
45473                 
45474                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45475                 ah -= 5; // knock a few pixes off for look..
45476 //                Roo.log(ah);
45477                 this.el.setHeight(this.adjustWidth('textarea', ah));
45478                 var eh = ah;
45479             }
45480         }
45481         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45482         this.editorcore.onResize(ew,eh);
45483         
45484     },
45485
45486     /**
45487      * Toggles the editor between standard and source edit mode.
45488      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45489      */
45490     toggleSourceEdit : function(sourceEditMode)
45491     {
45492         this.editorcore.toggleSourceEdit(sourceEditMode);
45493         
45494         if(this.editorcore.sourceEditMode){
45495             Roo.log('editor - showing textarea');
45496             
45497 //            Roo.log('in');
45498 //            Roo.log(this.syncValue());
45499             this.editorcore.syncValue();
45500             this.el.removeClass('x-hidden');
45501             this.el.dom.removeAttribute('tabIndex');
45502             this.el.focus();
45503             
45504             for (var i = 0; i < this.toolbars.length; i++) {
45505                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45506                     this.toolbars[i].tb.hide();
45507                     this.toolbars[i].footer.hide();
45508                 }
45509             }
45510             
45511         }else{
45512             Roo.log('editor - hiding textarea');
45513 //            Roo.log('out')
45514 //            Roo.log(this.pushValue()); 
45515             this.editorcore.pushValue();
45516             
45517             this.el.addClass('x-hidden');
45518             this.el.dom.setAttribute('tabIndex', -1);
45519             
45520             for (var i = 0; i < this.toolbars.length; i++) {
45521                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45522                     this.toolbars[i].tb.show();
45523                     this.toolbars[i].footer.show();
45524                 }
45525             }
45526             
45527             //this.deferFocus();
45528         }
45529         
45530         this.setSize(this.wrap.getSize());
45531         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45532         
45533         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45534     },
45535  
45536     // private (for BoxComponent)
45537     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45538
45539     // private (for BoxComponent)
45540     getResizeEl : function(){
45541         return this.wrap;
45542     },
45543
45544     // private (for BoxComponent)
45545     getPositionEl : function(){
45546         return this.wrap;
45547     },
45548
45549     // private
45550     initEvents : function(){
45551         this.originalValue = this.getValue();
45552     },
45553
45554     /**
45555      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45556      * @method
45557      */
45558     markInvalid : Roo.emptyFn,
45559     /**
45560      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45561      * @method
45562      */
45563     clearInvalid : Roo.emptyFn,
45564
45565     setValue : function(v){
45566         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45567         this.editorcore.pushValue();
45568     },
45569
45570      
45571     // private
45572     deferFocus : function(){
45573         this.focus.defer(10, this);
45574     },
45575
45576     // doc'ed in Field
45577     focus : function(){
45578         this.editorcore.focus();
45579         
45580     },
45581       
45582
45583     // private
45584     onDestroy : function(){
45585         
45586         
45587         
45588         if(this.rendered){
45589             
45590             for (var i =0; i < this.toolbars.length;i++) {
45591                 // fixme - ask toolbars for heights?
45592                 this.toolbars[i].onDestroy();
45593             }
45594             
45595             this.wrap.dom.innerHTML = '';
45596             this.wrap.remove();
45597         }
45598     },
45599
45600     // private
45601     onFirstFocus : function(){
45602         //Roo.log("onFirstFocus");
45603         this.editorcore.onFirstFocus();
45604          for (var i =0; i < this.toolbars.length;i++) {
45605             this.toolbars[i].onFirstFocus();
45606         }
45607         
45608     },
45609     
45610     // private
45611     syncValue : function()
45612     {
45613         this.editorcore.syncValue();
45614     },
45615     
45616     pushValue : function()
45617     {
45618         this.editorcore.pushValue();
45619     },
45620     
45621     setStylesheets : function(stylesheets)
45622     {
45623         this.editorcore.setStylesheets(stylesheets);
45624     },
45625     
45626     removeStylesheets : function()
45627     {
45628         this.editorcore.removeStylesheets();
45629     }
45630      
45631     
45632     // hide stuff that is not compatible
45633     /**
45634      * @event blur
45635      * @hide
45636      */
45637     /**
45638      * @event change
45639      * @hide
45640      */
45641     /**
45642      * @event focus
45643      * @hide
45644      */
45645     /**
45646      * @event specialkey
45647      * @hide
45648      */
45649     /**
45650      * @cfg {String} fieldClass @hide
45651      */
45652     /**
45653      * @cfg {String} focusClass @hide
45654      */
45655     /**
45656      * @cfg {String} autoCreate @hide
45657      */
45658     /**
45659      * @cfg {String} inputType @hide
45660      */
45661     /**
45662      * @cfg {String} invalidClass @hide
45663      */
45664     /**
45665      * @cfg {String} invalidText @hide
45666      */
45667     /**
45668      * @cfg {String} msgFx @hide
45669      */
45670     /**
45671      * @cfg {String} validateOnBlur @hide
45672      */
45673 });
45674  
45675     // <script type="text/javascript">
45676 /*
45677  * Based on
45678  * Ext JS Library 1.1.1
45679  * Copyright(c) 2006-2007, Ext JS, LLC.
45680  *  
45681  
45682  */
45683
45684 /**
45685  * @class Roo.form.HtmlEditorToolbar1
45686  * Basic Toolbar
45687  * 
45688  * Usage:
45689  *
45690  new Roo.form.HtmlEditor({
45691     ....
45692     toolbars : [
45693         new Roo.form.HtmlEditorToolbar1({
45694             disable : { fonts: 1 , format: 1, ..., ... , ...],
45695             btns : [ .... ]
45696         })
45697     }
45698      
45699  * 
45700  * @cfg {Object} disable List of elements to disable..
45701  * @cfg {Array} btns List of additional buttons.
45702  * 
45703  * 
45704  * NEEDS Extra CSS? 
45705  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45706  */
45707  
45708 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45709 {
45710     
45711     Roo.apply(this, config);
45712     
45713     // default disabled, based on 'good practice'..
45714     this.disable = this.disable || {};
45715     Roo.applyIf(this.disable, {
45716         fontSize : true,
45717         colors : true,
45718         specialElements : true
45719     });
45720     
45721     
45722     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45723     // dont call parent... till later.
45724 }
45725
45726 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45727     
45728     tb: false,
45729     
45730     rendered: false,
45731     
45732     editor : false,
45733     editorcore : false,
45734     /**
45735      * @cfg {Object} disable  List of toolbar elements to disable
45736          
45737      */
45738     disable : false,
45739     
45740     
45741      /**
45742      * @cfg {String} createLinkText The default text for the create link prompt
45743      */
45744     createLinkText : 'Please enter the URL for the link:',
45745     /**
45746      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45747      */
45748     defaultLinkValue : 'http:/'+'/',
45749    
45750     
45751       /**
45752      * @cfg {Array} fontFamilies An array of available font families
45753      */
45754     fontFamilies : [
45755         'Arial',
45756         'Courier New',
45757         'Tahoma',
45758         'Times New Roman',
45759         'Verdana'
45760     ],
45761     
45762     specialChars : [
45763            "&#169;",
45764           "&#174;",     
45765           "&#8482;",    
45766           "&#163;" ,    
45767          // "&#8212;",    
45768           "&#8230;",    
45769           "&#247;" ,    
45770         //  "&#225;" ,     ?? a acute?
45771            "&#8364;"    , //Euro
45772        //   "&#8220;"    ,
45773         //  "&#8221;"    ,
45774         //  "&#8226;"    ,
45775           "&#176;"  //   , // degrees
45776
45777          // "&#233;"     , // e ecute
45778          // "&#250;"     , // u ecute?
45779     ],
45780     
45781     specialElements : [
45782         {
45783             text: "Insert Table",
45784             xtype: 'MenuItem',
45785             xns : Roo.Menu,
45786             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45787                 
45788         },
45789         {    
45790             text: "Insert Image",
45791             xtype: 'MenuItem',
45792             xns : Roo.Menu,
45793             ihtml : '<img src="about:blank"/>'
45794             
45795         }
45796         
45797          
45798     ],
45799     
45800     
45801     inputElements : [ 
45802             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45803             "input:submit", "input:button", "select", "textarea", "label" ],
45804     formats : [
45805         ["p"] ,  
45806         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45807         ["pre"],[ "code"], 
45808         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45809         ['div'],['span'],
45810         ['sup'],['sub']
45811     ],
45812     
45813     cleanStyles : [
45814         "font-size"
45815     ],
45816      /**
45817      * @cfg {String} defaultFont default font to use.
45818      */
45819     defaultFont: 'tahoma',
45820    
45821     fontSelect : false,
45822     
45823     
45824     formatCombo : false,
45825     
45826     init : function(editor)
45827     {
45828         this.editor = editor;
45829         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45830         var editorcore = this.editorcore;
45831         
45832         var _t = this;
45833         
45834         var fid = editorcore.frameId;
45835         var etb = this;
45836         function btn(id, toggle, handler){
45837             var xid = fid + '-'+ id ;
45838             return {
45839                 id : xid,
45840                 cmd : id,
45841                 cls : 'x-btn-icon x-edit-'+id,
45842                 enableToggle:toggle !== false,
45843                 scope: _t, // was editor...
45844                 handler:handler||_t.relayBtnCmd,
45845                 clickEvent:'mousedown',
45846                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45847                 tabIndex:-1
45848             };
45849         }
45850         
45851         
45852         
45853         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45854         this.tb = tb;
45855          // stop form submits
45856         tb.el.on('click', function(e){
45857             e.preventDefault(); // what does this do?
45858         });
45859
45860         if(!this.disable.font) { // && !Roo.isSafari){
45861             /* why no safari for fonts 
45862             editor.fontSelect = tb.el.createChild({
45863                 tag:'select',
45864                 tabIndex: -1,
45865                 cls:'x-font-select',
45866                 html: this.createFontOptions()
45867             });
45868             
45869             editor.fontSelect.on('change', function(){
45870                 var font = editor.fontSelect.dom.value;
45871                 editor.relayCmd('fontname', font);
45872                 editor.deferFocus();
45873             }, editor);
45874             
45875             tb.add(
45876                 editor.fontSelect.dom,
45877                 '-'
45878             );
45879             */
45880             
45881         };
45882         if(!this.disable.formats){
45883             this.formatCombo = new Roo.form.ComboBox({
45884                 store: new Roo.data.SimpleStore({
45885                     id : 'tag',
45886                     fields: ['tag'],
45887                     data : this.formats // from states.js
45888                 }),
45889                 blockFocus : true,
45890                 name : '',
45891                 //autoCreate : {tag: "div",  size: "20"},
45892                 displayField:'tag',
45893                 typeAhead: false,
45894                 mode: 'local',
45895                 editable : false,
45896                 triggerAction: 'all',
45897                 emptyText:'Add tag',
45898                 selectOnFocus:true,
45899                 width:135,
45900                 listeners : {
45901                     'select': function(c, r, i) {
45902                         editorcore.insertTag(r.get('tag'));
45903                         editor.focus();
45904                     }
45905                 }
45906
45907             });
45908             tb.addField(this.formatCombo);
45909             
45910         }
45911         
45912         if(!this.disable.format){
45913             tb.add(
45914                 btn('bold'),
45915                 btn('italic'),
45916                 btn('underline'),
45917                 btn('strikethrough')
45918             );
45919         };
45920         if(!this.disable.fontSize){
45921             tb.add(
45922                 '-',
45923                 
45924                 
45925                 btn('increasefontsize', false, editorcore.adjustFont),
45926                 btn('decreasefontsize', false, editorcore.adjustFont)
45927             );
45928         };
45929         
45930         
45931         if(!this.disable.colors){
45932             tb.add(
45933                 '-', {
45934                     id:editorcore.frameId +'-forecolor',
45935                     cls:'x-btn-icon x-edit-forecolor',
45936                     clickEvent:'mousedown',
45937                     tooltip: this.buttonTips['forecolor'] || undefined,
45938                     tabIndex:-1,
45939                     menu : new Roo.menu.ColorMenu({
45940                         allowReselect: true,
45941                         focus: Roo.emptyFn,
45942                         value:'000000',
45943                         plain:true,
45944                         selectHandler: function(cp, color){
45945                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45946                             editor.deferFocus();
45947                         },
45948                         scope: editorcore,
45949                         clickEvent:'mousedown'
45950                     })
45951                 }, {
45952                     id:editorcore.frameId +'backcolor',
45953                     cls:'x-btn-icon x-edit-backcolor',
45954                     clickEvent:'mousedown',
45955                     tooltip: this.buttonTips['backcolor'] || undefined,
45956                     tabIndex:-1,
45957                     menu : new Roo.menu.ColorMenu({
45958                         focus: Roo.emptyFn,
45959                         value:'FFFFFF',
45960                         plain:true,
45961                         allowReselect: true,
45962                         selectHandler: function(cp, color){
45963                             if(Roo.isGecko){
45964                                 editorcore.execCmd('useCSS', false);
45965                                 editorcore.execCmd('hilitecolor', color);
45966                                 editorcore.execCmd('useCSS', true);
45967                                 editor.deferFocus();
45968                             }else{
45969                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45970                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45971                                 editor.deferFocus();
45972                             }
45973                         },
45974                         scope:editorcore,
45975                         clickEvent:'mousedown'
45976                     })
45977                 }
45978             );
45979         };
45980         // now add all the items...
45981         
45982
45983         if(!this.disable.alignments){
45984             tb.add(
45985                 '-',
45986                 btn('justifyleft'),
45987                 btn('justifycenter'),
45988                 btn('justifyright')
45989             );
45990         };
45991
45992         //if(!Roo.isSafari){
45993             if(!this.disable.links){
45994                 tb.add(
45995                     '-',
45996                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45997                 );
45998             };
45999
46000             if(!this.disable.lists){
46001                 tb.add(
46002                     '-',
46003                     btn('insertorderedlist'),
46004                     btn('insertunorderedlist')
46005                 );
46006             }
46007             if(!this.disable.sourceEdit){
46008                 tb.add(
46009                     '-',
46010                     btn('sourceedit', true, function(btn){
46011                         this.toggleSourceEdit(btn.pressed);
46012                     })
46013                 );
46014             }
46015         //}
46016         
46017         var smenu = { };
46018         // special menu.. - needs to be tidied up..
46019         if (!this.disable.special) {
46020             smenu = {
46021                 text: "&#169;",
46022                 cls: 'x-edit-none',
46023                 
46024                 menu : {
46025                     items : []
46026                 }
46027             };
46028             for (var i =0; i < this.specialChars.length; i++) {
46029                 smenu.menu.items.push({
46030                     
46031                     html: this.specialChars[i],
46032                     handler: function(a,b) {
46033                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46034                         //editor.insertAtCursor(a.html);
46035                         
46036                     },
46037                     tabIndex:-1
46038                 });
46039             }
46040             
46041             
46042             tb.add(smenu);
46043             
46044             
46045         }
46046         
46047         var cmenu = { };
46048         if (!this.disable.cleanStyles) {
46049             cmenu = {
46050                 cls: 'x-btn-icon x-btn-clear',
46051                 
46052                 menu : {
46053                     items : []
46054                 }
46055             };
46056             for (var i =0; i < this.cleanStyles.length; i++) {
46057                 cmenu.menu.items.push({
46058                     actiontype : this.cleanStyles[i],
46059                     html: 'Remove ' + this.cleanStyles[i],
46060                     handler: function(a,b) {
46061 //                        Roo.log(a);
46062 //                        Roo.log(b);
46063                         var c = Roo.get(editorcore.doc.body);
46064                         c.select('[style]').each(function(s) {
46065                             s.dom.style.removeProperty(a.actiontype);
46066                         });
46067                         editorcore.syncValue();
46068                     },
46069                     tabIndex:-1
46070                 });
46071             }
46072              cmenu.menu.items.push({
46073                 actiontype : 'tablewidths',
46074                 html: 'Remove Table Widths',
46075                 handler: function(a,b) {
46076                     editorcore.cleanTableWidths();
46077                     editorcore.syncValue();
46078                 },
46079                 tabIndex:-1
46080             });
46081             cmenu.menu.items.push({
46082                 actiontype : 'word',
46083                 html: 'Remove MS Word Formating',
46084                 handler: function(a,b) {
46085                     editorcore.cleanWord();
46086                     editorcore.syncValue();
46087                 },
46088                 tabIndex:-1
46089             });
46090             
46091             cmenu.menu.items.push({
46092                 actiontype : 'all',
46093                 html: 'Remove All Styles',
46094                 handler: function(a,b) {
46095                     
46096                     var c = Roo.get(editorcore.doc.body);
46097                     c.select('[style]').each(function(s) {
46098                         s.dom.removeAttribute('style');
46099                     });
46100                     editorcore.syncValue();
46101                 },
46102                 tabIndex:-1
46103             });
46104             
46105             cmenu.menu.items.push({
46106                 actiontype : 'all',
46107                 html: 'Remove All CSS Classes',
46108                 handler: function(a,b) {
46109                     
46110                     var c = Roo.get(editorcore.doc.body);
46111                     c.select('[class]').each(function(s) {
46112                         s.dom.removeAttribute('class');
46113                     });
46114                     editorcore.cleanWord();
46115                     editorcore.syncValue();
46116                 },
46117                 tabIndex:-1
46118             });
46119             
46120              cmenu.menu.items.push({
46121                 actiontype : 'tidy',
46122                 html: 'Tidy HTML Source',
46123                 handler: function(a,b) {
46124                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46125                     editorcore.syncValue();
46126                 },
46127                 tabIndex:-1
46128             });
46129             
46130             
46131             tb.add(cmenu);
46132         }
46133          
46134         if (!this.disable.specialElements) {
46135             var semenu = {
46136                 text: "Other;",
46137                 cls: 'x-edit-none',
46138                 menu : {
46139                     items : []
46140                 }
46141             };
46142             for (var i =0; i < this.specialElements.length; i++) {
46143                 semenu.menu.items.push(
46144                     Roo.apply({ 
46145                         handler: function(a,b) {
46146                             editor.insertAtCursor(this.ihtml);
46147                         }
46148                     }, this.specialElements[i])
46149                 );
46150                     
46151             }
46152             
46153             tb.add(semenu);
46154             
46155             
46156         }
46157          
46158         
46159         if (this.btns) {
46160             for(var i =0; i< this.btns.length;i++) {
46161                 var b = Roo.factory(this.btns[i],Roo.form);
46162                 b.cls =  'x-edit-none';
46163                 
46164                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46165                     b.cls += ' x-init-enable';
46166                 }
46167                 
46168                 b.scope = editorcore;
46169                 tb.add(b);
46170             }
46171         
46172         }
46173         
46174         
46175         
46176         // disable everything...
46177         
46178         this.tb.items.each(function(item){
46179             
46180            if(
46181                 item.id != editorcore.frameId+ '-sourceedit' && 
46182                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46183             ){
46184                 
46185                 item.disable();
46186             }
46187         });
46188         this.rendered = true;
46189         
46190         // the all the btns;
46191         editor.on('editorevent', this.updateToolbar, this);
46192         // other toolbars need to implement this..
46193         //editor.on('editmodechange', this.updateToolbar, this);
46194     },
46195     
46196     
46197     relayBtnCmd : function(btn) {
46198         this.editorcore.relayCmd(btn.cmd);
46199     },
46200     // private used internally
46201     createLink : function(){
46202         Roo.log("create link?");
46203         var url = prompt(this.createLinkText, this.defaultLinkValue);
46204         if(url && url != 'http:/'+'/'){
46205             this.editorcore.relayCmd('createlink', url);
46206         }
46207     },
46208
46209     
46210     /**
46211      * Protected method that will not generally be called directly. It triggers
46212      * a toolbar update by reading the markup state of the current selection in the editor.
46213      */
46214     updateToolbar: function(){
46215
46216         if(!this.editorcore.activated){
46217             this.editor.onFirstFocus();
46218             return;
46219         }
46220
46221         var btns = this.tb.items.map, 
46222             doc = this.editorcore.doc,
46223             frameId = this.editorcore.frameId;
46224
46225         if(!this.disable.font && !Roo.isSafari){
46226             /*
46227             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46228             if(name != this.fontSelect.dom.value){
46229                 this.fontSelect.dom.value = name;
46230             }
46231             */
46232         }
46233         if(!this.disable.format){
46234             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46235             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46236             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46237             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46238         }
46239         if(!this.disable.alignments){
46240             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46241             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46242             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46243         }
46244         if(!Roo.isSafari && !this.disable.lists){
46245             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46246             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46247         }
46248         
46249         var ans = this.editorcore.getAllAncestors();
46250         if (this.formatCombo) {
46251             
46252             
46253             var store = this.formatCombo.store;
46254             this.formatCombo.setValue("");
46255             for (var i =0; i < ans.length;i++) {
46256                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46257                     // select it..
46258                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46259                     break;
46260                 }
46261             }
46262         }
46263         
46264         
46265         
46266         // hides menus... - so this cant be on a menu...
46267         Roo.menu.MenuMgr.hideAll();
46268
46269         //this.editorsyncValue();
46270     },
46271    
46272     
46273     createFontOptions : function(){
46274         var buf = [], fs = this.fontFamilies, ff, lc;
46275         
46276         
46277         
46278         for(var i = 0, len = fs.length; i< len; i++){
46279             ff = fs[i];
46280             lc = ff.toLowerCase();
46281             buf.push(
46282                 '<option value="',lc,'" style="font-family:',ff,';"',
46283                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46284                     ff,
46285                 '</option>'
46286             );
46287         }
46288         return buf.join('');
46289     },
46290     
46291     toggleSourceEdit : function(sourceEditMode){
46292         
46293         Roo.log("toolbar toogle");
46294         if(sourceEditMode === undefined){
46295             sourceEditMode = !this.sourceEditMode;
46296         }
46297         this.sourceEditMode = sourceEditMode === true;
46298         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46299         // just toggle the button?
46300         if(btn.pressed !== this.sourceEditMode){
46301             btn.toggle(this.sourceEditMode);
46302             return;
46303         }
46304         
46305         if(sourceEditMode){
46306             Roo.log("disabling buttons");
46307             this.tb.items.each(function(item){
46308                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46309                     item.disable();
46310                 }
46311             });
46312           
46313         }else{
46314             Roo.log("enabling buttons");
46315             if(this.editorcore.initialized){
46316                 this.tb.items.each(function(item){
46317                     item.enable();
46318                 });
46319             }
46320             
46321         }
46322         Roo.log("calling toggole on editor");
46323         // tell the editor that it's been pressed..
46324         this.editor.toggleSourceEdit(sourceEditMode);
46325        
46326     },
46327      /**
46328      * Object collection of toolbar tooltips for the buttons in the editor. The key
46329      * is the command id associated with that button and the value is a valid QuickTips object.
46330      * For example:
46331 <pre><code>
46332 {
46333     bold : {
46334         title: 'Bold (Ctrl+B)',
46335         text: 'Make the selected text bold.',
46336         cls: 'x-html-editor-tip'
46337     },
46338     italic : {
46339         title: 'Italic (Ctrl+I)',
46340         text: 'Make the selected text italic.',
46341         cls: 'x-html-editor-tip'
46342     },
46343     ...
46344 </code></pre>
46345     * @type Object
46346      */
46347     buttonTips : {
46348         bold : {
46349             title: 'Bold (Ctrl+B)',
46350             text: 'Make the selected text bold.',
46351             cls: 'x-html-editor-tip'
46352         },
46353         italic : {
46354             title: 'Italic (Ctrl+I)',
46355             text: 'Make the selected text italic.',
46356             cls: 'x-html-editor-tip'
46357         },
46358         underline : {
46359             title: 'Underline (Ctrl+U)',
46360             text: 'Underline the selected text.',
46361             cls: 'x-html-editor-tip'
46362         },
46363         strikethrough : {
46364             title: 'Strikethrough',
46365             text: 'Strikethrough the selected text.',
46366             cls: 'x-html-editor-tip'
46367         },
46368         increasefontsize : {
46369             title: 'Grow Text',
46370             text: 'Increase the font size.',
46371             cls: 'x-html-editor-tip'
46372         },
46373         decreasefontsize : {
46374             title: 'Shrink Text',
46375             text: 'Decrease the font size.',
46376             cls: 'x-html-editor-tip'
46377         },
46378         backcolor : {
46379             title: 'Text Highlight Color',
46380             text: 'Change the background color of the selected text.',
46381             cls: 'x-html-editor-tip'
46382         },
46383         forecolor : {
46384             title: 'Font Color',
46385             text: 'Change the color of the selected text.',
46386             cls: 'x-html-editor-tip'
46387         },
46388         justifyleft : {
46389             title: 'Align Text Left',
46390             text: 'Align text to the left.',
46391             cls: 'x-html-editor-tip'
46392         },
46393         justifycenter : {
46394             title: 'Center Text',
46395             text: 'Center text in the editor.',
46396             cls: 'x-html-editor-tip'
46397         },
46398         justifyright : {
46399             title: 'Align Text Right',
46400             text: 'Align text to the right.',
46401             cls: 'x-html-editor-tip'
46402         },
46403         insertunorderedlist : {
46404             title: 'Bullet List',
46405             text: 'Start a bulleted list.',
46406             cls: 'x-html-editor-tip'
46407         },
46408         insertorderedlist : {
46409             title: 'Numbered List',
46410             text: 'Start a numbered list.',
46411             cls: 'x-html-editor-tip'
46412         },
46413         createlink : {
46414             title: 'Hyperlink',
46415             text: 'Make the selected text a hyperlink.',
46416             cls: 'x-html-editor-tip'
46417         },
46418         sourceedit : {
46419             title: 'Source Edit',
46420             text: 'Switch to source editing mode.',
46421             cls: 'x-html-editor-tip'
46422         }
46423     },
46424     // private
46425     onDestroy : function(){
46426         if(this.rendered){
46427             
46428             this.tb.items.each(function(item){
46429                 if(item.menu){
46430                     item.menu.removeAll();
46431                     if(item.menu.el){
46432                         item.menu.el.destroy();
46433                     }
46434                 }
46435                 item.destroy();
46436             });
46437              
46438         }
46439     },
46440     onFirstFocus: function() {
46441         this.tb.items.each(function(item){
46442            item.enable();
46443         });
46444     }
46445 });
46446
46447
46448
46449
46450 // <script type="text/javascript">
46451 /*
46452  * Based on
46453  * Ext JS Library 1.1.1
46454  * Copyright(c) 2006-2007, Ext JS, LLC.
46455  *  
46456  
46457  */
46458
46459  
46460 /**
46461  * @class Roo.form.HtmlEditor.ToolbarContext
46462  * Context Toolbar
46463  * 
46464  * Usage:
46465  *
46466  new Roo.form.HtmlEditor({
46467     ....
46468     toolbars : [
46469         { xtype: 'ToolbarStandard', styles : {} }
46470         { xtype: 'ToolbarContext', disable : {} }
46471     ]
46472 })
46473
46474      
46475  * 
46476  * @config : {Object} disable List of elements to disable.. (not done yet.)
46477  * @config : {Object} styles  Map of styles available.
46478  * 
46479  */
46480
46481 Roo.form.HtmlEditor.ToolbarContext = function(config)
46482 {
46483     
46484     Roo.apply(this, config);
46485     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46486     // dont call parent... till later.
46487     this.styles = this.styles || {};
46488 }
46489
46490  
46491
46492 Roo.form.HtmlEditor.ToolbarContext.types = {
46493     'IMG' : {
46494         width : {
46495             title: "Width",
46496             width: 40
46497         },
46498         height:  {
46499             title: "Height",
46500             width: 40
46501         },
46502         align: {
46503             title: "Align",
46504             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46505             width : 80
46506             
46507         },
46508         border: {
46509             title: "Border",
46510             width: 40
46511         },
46512         alt: {
46513             title: "Alt",
46514             width: 120
46515         },
46516         src : {
46517             title: "Src",
46518             width: 220
46519         }
46520         
46521     },
46522     'A' : {
46523         name : {
46524             title: "Name",
46525             width: 50
46526         },
46527         target:  {
46528             title: "Target",
46529             width: 120
46530         },
46531         href:  {
46532             title: "Href",
46533             width: 220
46534         } // border?
46535         
46536     },
46537     'TABLE' : {
46538         rows : {
46539             title: "Rows",
46540             width: 20
46541         },
46542         cols : {
46543             title: "Cols",
46544             width: 20
46545         },
46546         width : {
46547             title: "Width",
46548             width: 40
46549         },
46550         height : {
46551             title: "Height",
46552             width: 40
46553         },
46554         border : {
46555             title: "Border",
46556             width: 20
46557         }
46558     },
46559     'TD' : {
46560         width : {
46561             title: "Width",
46562             width: 40
46563         },
46564         height : {
46565             title: "Height",
46566             width: 40
46567         },   
46568         align: {
46569             title: "Align",
46570             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46571             width: 80
46572         },
46573         valign: {
46574             title: "Valign",
46575             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46576             width: 80
46577         },
46578         colspan: {
46579             title: "Colspan",
46580             width: 20
46581             
46582         },
46583          'font-family'  : {
46584             title : "Font",
46585             style : 'fontFamily',
46586             displayField: 'display',
46587             optname : 'font-family',
46588             width: 140
46589         }
46590     },
46591     'INPUT' : {
46592         name : {
46593             title: "name",
46594             width: 120
46595         },
46596         value : {
46597             title: "Value",
46598             width: 120
46599         },
46600         width : {
46601             title: "Width",
46602             width: 40
46603         }
46604     },
46605     'LABEL' : {
46606         'for' : {
46607             title: "For",
46608             width: 120
46609         }
46610     },
46611     'TEXTAREA' : {
46612           name : {
46613             title: "name",
46614             width: 120
46615         },
46616         rows : {
46617             title: "Rows",
46618             width: 20
46619         },
46620         cols : {
46621             title: "Cols",
46622             width: 20
46623         }
46624     },
46625     'SELECT' : {
46626         name : {
46627             title: "name",
46628             width: 120
46629         },
46630         selectoptions : {
46631             title: "Options",
46632             width: 200
46633         }
46634     },
46635     
46636     // should we really allow this??
46637     // should this just be 
46638     'BODY' : {
46639         title : {
46640             title: "Title",
46641             width: 200,
46642             disabled : true
46643         }
46644     },
46645     'SPAN' : {
46646         'font-family'  : {
46647             title : "Font",
46648             style : 'fontFamily',
46649             displayField: 'display',
46650             optname : 'font-family',
46651             width: 140
46652         }
46653     },
46654     'DIV' : {
46655         'font-family'  : {
46656             title : "Font",
46657             style : 'fontFamily',
46658             displayField: 'display',
46659             optname : 'font-family',
46660             width: 140
46661         }
46662     },
46663      'P' : {
46664         'font-family'  : {
46665             title : "Font",
46666             style : 'fontFamily',
46667             displayField: 'display',
46668             optname : 'font-family',
46669             width: 140
46670         }
46671     },
46672     
46673     '*' : {
46674         // empty..
46675     }
46676
46677 };
46678
46679 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46680 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46681
46682 Roo.form.HtmlEditor.ToolbarContext.options = {
46683         'font-family'  : [ 
46684                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46685                 [ 'Courier New', 'Courier New'],
46686                 [ 'Tahoma', 'Tahoma'],
46687                 [ 'Times New Roman,serif', 'Times'],
46688                 [ 'Verdana','Verdana' ]
46689         ]
46690 };
46691
46692 // fixme - these need to be configurable..
46693  
46694
46695 //Roo.form.HtmlEditor.ToolbarContext.types
46696
46697
46698 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46699     
46700     tb: false,
46701     
46702     rendered: false,
46703     
46704     editor : false,
46705     editorcore : false,
46706     /**
46707      * @cfg {Object} disable  List of toolbar elements to disable
46708          
46709      */
46710     disable : false,
46711     /**
46712      * @cfg {Object} styles List of styles 
46713      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46714      *
46715      * These must be defined in the page, so they get rendered correctly..
46716      * .headline { }
46717      * TD.underline { }
46718      * 
46719      */
46720     styles : false,
46721     
46722     options: false,
46723     
46724     toolbars : false,
46725     
46726     init : function(editor)
46727     {
46728         this.editor = editor;
46729         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46730         var editorcore = this.editorcore;
46731         
46732         var fid = editorcore.frameId;
46733         var etb = this;
46734         function btn(id, toggle, handler){
46735             var xid = fid + '-'+ id ;
46736             return {
46737                 id : xid,
46738                 cmd : id,
46739                 cls : 'x-btn-icon x-edit-'+id,
46740                 enableToggle:toggle !== false,
46741                 scope: editorcore, // was editor...
46742                 handler:handler||editorcore.relayBtnCmd,
46743                 clickEvent:'mousedown',
46744                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46745                 tabIndex:-1
46746             };
46747         }
46748         // create a new element.
46749         var wdiv = editor.wrap.createChild({
46750                 tag: 'div'
46751             }, editor.wrap.dom.firstChild.nextSibling, true);
46752         
46753         // can we do this more than once??
46754         
46755          // stop form submits
46756       
46757  
46758         // disable everything...
46759         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46760         this.toolbars = {};
46761            
46762         for (var i in  ty) {
46763           
46764             this.toolbars[i] = this.buildToolbar(ty[i],i);
46765         }
46766         this.tb = this.toolbars.BODY;
46767         this.tb.el.show();
46768         this.buildFooter();
46769         this.footer.show();
46770         editor.on('hide', function( ) { this.footer.hide() }, this);
46771         editor.on('show', function( ) { this.footer.show() }, this);
46772         
46773          
46774         this.rendered = true;
46775         
46776         // the all the btns;
46777         editor.on('editorevent', this.updateToolbar, this);
46778         // other toolbars need to implement this..
46779         //editor.on('editmodechange', this.updateToolbar, this);
46780     },
46781     
46782     
46783     
46784     /**
46785      * Protected method that will not generally be called directly. It triggers
46786      * a toolbar update by reading the markup state of the current selection in the editor.
46787      *
46788      * Note you can force an update by calling on('editorevent', scope, false)
46789      */
46790     updateToolbar: function(editor,ev,sel){
46791
46792         //Roo.log(ev);
46793         // capture mouse up - this is handy for selecting images..
46794         // perhaps should go somewhere else...
46795         if(!this.editorcore.activated){
46796              this.editor.onFirstFocus();
46797             return;
46798         }
46799         
46800         
46801         
46802         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46803         // selectNode - might want to handle IE?
46804         if (ev &&
46805             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46806             ev.target && ev.target.tagName == 'IMG') {
46807             // they have click on an image...
46808             // let's see if we can change the selection...
46809             sel = ev.target;
46810          
46811               var nodeRange = sel.ownerDocument.createRange();
46812             try {
46813                 nodeRange.selectNode(sel);
46814             } catch (e) {
46815                 nodeRange.selectNodeContents(sel);
46816             }
46817             //nodeRange.collapse(true);
46818             var s = this.editorcore.win.getSelection();
46819             s.removeAllRanges();
46820             s.addRange(nodeRange);
46821         }  
46822         
46823       
46824         var updateFooter = sel ? false : true;
46825         
46826         
46827         var ans = this.editorcore.getAllAncestors();
46828         
46829         // pick
46830         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46831         
46832         if (!sel) { 
46833             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46834             sel = sel ? sel : this.editorcore.doc.body;
46835             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46836             
46837         }
46838         // pick a menu that exists..
46839         var tn = sel.tagName.toUpperCase();
46840         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46841         
46842         tn = sel.tagName.toUpperCase();
46843         
46844         var lastSel = this.tb.selectedNode;
46845         
46846         this.tb.selectedNode = sel;
46847         
46848         // if current menu does not match..
46849         
46850         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46851                 
46852             this.tb.el.hide();
46853             ///console.log("show: " + tn);
46854             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46855             this.tb.el.show();
46856             // update name
46857             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46858             
46859             
46860             // update attributes
46861             if (this.tb.fields) {
46862                 this.tb.fields.each(function(e) {
46863                     if (e.stylename) {
46864                         e.setValue(sel.style[e.stylename]);
46865                         return;
46866                     } 
46867                    e.setValue(sel.getAttribute(e.attrname));
46868                 });
46869             }
46870             
46871             var hasStyles = false;
46872             for(var i in this.styles) {
46873                 hasStyles = true;
46874                 break;
46875             }
46876             
46877             // update styles
46878             if (hasStyles) { 
46879                 var st = this.tb.fields.item(0);
46880                 
46881                 st.store.removeAll();
46882                
46883                 
46884                 var cn = sel.className.split(/\s+/);
46885                 
46886                 var avs = [];
46887                 if (this.styles['*']) {
46888                     
46889                     Roo.each(this.styles['*'], function(v) {
46890                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46891                     });
46892                 }
46893                 if (this.styles[tn]) { 
46894                     Roo.each(this.styles[tn], function(v) {
46895                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46896                     });
46897                 }
46898                 
46899                 st.store.loadData(avs);
46900                 st.collapse();
46901                 st.setValue(cn);
46902             }
46903             // flag our selected Node.
46904             this.tb.selectedNode = sel;
46905            
46906            
46907             Roo.menu.MenuMgr.hideAll();
46908
46909         }
46910         
46911         if (!updateFooter) {
46912             //this.footDisp.dom.innerHTML = ''; 
46913             return;
46914         }
46915         // update the footer
46916         //
46917         var html = '';
46918         
46919         this.footerEls = ans.reverse();
46920         Roo.each(this.footerEls, function(a,i) {
46921             if (!a) { return; }
46922             html += html.length ? ' &gt; '  :  '';
46923             
46924             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46925             
46926         });
46927        
46928         // 
46929         var sz = this.footDisp.up('td').getSize();
46930         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46931         this.footDisp.dom.style.marginLeft = '5px';
46932         
46933         this.footDisp.dom.style.overflow = 'hidden';
46934         
46935         this.footDisp.dom.innerHTML = html;
46936             
46937         //this.editorsyncValue();
46938     },
46939      
46940     
46941    
46942        
46943     // private
46944     onDestroy : function(){
46945         if(this.rendered){
46946             
46947             this.tb.items.each(function(item){
46948                 if(item.menu){
46949                     item.menu.removeAll();
46950                     if(item.menu.el){
46951                         item.menu.el.destroy();
46952                     }
46953                 }
46954                 item.destroy();
46955             });
46956              
46957         }
46958     },
46959     onFirstFocus: function() {
46960         // need to do this for all the toolbars..
46961         this.tb.items.each(function(item){
46962            item.enable();
46963         });
46964     },
46965     buildToolbar: function(tlist, nm)
46966     {
46967         var editor = this.editor;
46968         var editorcore = this.editorcore;
46969          // create a new element.
46970         var wdiv = editor.wrap.createChild({
46971                 tag: 'div'
46972             }, editor.wrap.dom.firstChild.nextSibling, true);
46973         
46974        
46975         var tb = new Roo.Toolbar(wdiv);
46976         // add the name..
46977         
46978         tb.add(nm+ ":&nbsp;");
46979         
46980         var styles = [];
46981         for(var i in this.styles) {
46982             styles.push(i);
46983         }
46984         
46985         // styles...
46986         if (styles && styles.length) {
46987             
46988             // this needs a multi-select checkbox...
46989             tb.addField( new Roo.form.ComboBox({
46990                 store: new Roo.data.SimpleStore({
46991                     id : 'val',
46992                     fields: ['val', 'selected'],
46993                     data : [] 
46994                 }),
46995                 name : '-roo-edit-className',
46996                 attrname : 'className',
46997                 displayField: 'val',
46998                 typeAhead: false,
46999                 mode: 'local',
47000                 editable : false,
47001                 triggerAction: 'all',
47002                 emptyText:'Select Style',
47003                 selectOnFocus:true,
47004                 width: 130,
47005                 listeners : {
47006                     'select': function(c, r, i) {
47007                         // initial support only for on class per el..
47008                         tb.selectedNode.className =  r ? r.get('val') : '';
47009                         editorcore.syncValue();
47010                     }
47011                 }
47012     
47013             }));
47014         }
47015         
47016         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47017         var tbops = tbc.options;
47018         
47019         for (var i in tlist) {
47020             
47021             var item = tlist[i];
47022             tb.add(item.title + ":&nbsp;");
47023             
47024             
47025             //optname == used so you can configure the options available..
47026             var opts = item.opts ? item.opts : false;
47027             if (item.optname) {
47028                 opts = tbops[item.optname];
47029            
47030             }
47031             
47032             if (opts) {
47033                 // opts == pulldown..
47034                 tb.addField( new Roo.form.ComboBox({
47035                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47036                         id : 'val',
47037                         fields: ['val', 'display'],
47038                         data : opts  
47039                     }),
47040                     name : '-roo-edit-' + i,
47041                     attrname : i,
47042                     stylename : item.style ? item.style : false,
47043                     displayField: item.displayField ? item.displayField : 'val',
47044                     valueField :  'val',
47045                     typeAhead: false,
47046                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47047                     editable : false,
47048                     triggerAction: 'all',
47049                     emptyText:'Select',
47050                     selectOnFocus:true,
47051                     width: item.width ? item.width  : 130,
47052                     listeners : {
47053                         'select': function(c, r, i) {
47054                             if (c.stylename) {
47055                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47056                                 return;
47057                             }
47058                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47059                         }
47060                     }
47061
47062                 }));
47063                 continue;
47064                     
47065                  
47066                 
47067                 tb.addField( new Roo.form.TextField({
47068                     name: i,
47069                     width: 100,
47070                     //allowBlank:false,
47071                     value: ''
47072                 }));
47073                 continue;
47074             }
47075             tb.addField( new Roo.form.TextField({
47076                 name: '-roo-edit-' + i,
47077                 attrname : i,
47078                 
47079                 width: item.width,
47080                 //allowBlank:true,
47081                 value: '',
47082                 listeners: {
47083                     'change' : function(f, nv, ov) {
47084                         tb.selectedNode.setAttribute(f.attrname, nv);
47085                         editorcore.syncValue();
47086                     }
47087                 }
47088             }));
47089              
47090         }
47091         
47092         var _this = this;
47093         
47094         if(nm == 'BODY'){
47095             tb.addSeparator();
47096         
47097             tb.addButton( {
47098                 text: 'Stylesheets',
47099
47100                 listeners : {
47101                     click : function ()
47102                     {
47103                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47104                     }
47105                 }
47106             });
47107         }
47108         
47109         tb.addFill();
47110         tb.addButton( {
47111             text: 'Remove Tag',
47112     
47113             listeners : {
47114                 click : function ()
47115                 {
47116                     // remove
47117                     // undo does not work.
47118                      
47119                     var sn = tb.selectedNode;
47120                     
47121                     var pn = sn.parentNode;
47122                     
47123                     var stn =  sn.childNodes[0];
47124                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47125                     while (sn.childNodes.length) {
47126                         var node = sn.childNodes[0];
47127                         sn.removeChild(node);
47128                         //Roo.log(node);
47129                         pn.insertBefore(node, sn);
47130                         
47131                     }
47132                     pn.removeChild(sn);
47133                     var range = editorcore.createRange();
47134         
47135                     range.setStart(stn,0);
47136                     range.setEnd(en,0); //????
47137                     //range.selectNode(sel);
47138                     
47139                     
47140                     var selection = editorcore.getSelection();
47141                     selection.removeAllRanges();
47142                     selection.addRange(range);
47143                     
47144                     
47145                     
47146                     //_this.updateToolbar(null, null, pn);
47147                     _this.updateToolbar(null, null, null);
47148                     _this.footDisp.dom.innerHTML = ''; 
47149                 }
47150             }
47151             
47152                     
47153                 
47154             
47155         });
47156         
47157         
47158         tb.el.on('click', function(e){
47159             e.preventDefault(); // what does this do?
47160         });
47161         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47162         tb.el.hide();
47163         tb.name = nm;
47164         // dont need to disable them... as they will get hidden
47165         return tb;
47166          
47167         
47168     },
47169     buildFooter : function()
47170     {
47171         
47172         var fel = this.editor.wrap.createChild();
47173         this.footer = new Roo.Toolbar(fel);
47174         // toolbar has scrolly on left / right?
47175         var footDisp= new Roo.Toolbar.Fill();
47176         var _t = this;
47177         this.footer.add(
47178             {
47179                 text : '&lt;',
47180                 xtype: 'Button',
47181                 handler : function() {
47182                     _t.footDisp.scrollTo('left',0,true)
47183                 }
47184             }
47185         );
47186         this.footer.add( footDisp );
47187         this.footer.add( 
47188             {
47189                 text : '&gt;',
47190                 xtype: 'Button',
47191                 handler : function() {
47192                     // no animation..
47193                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47194                 }
47195             }
47196         );
47197         var fel = Roo.get(footDisp.el);
47198         fel.addClass('x-editor-context');
47199         this.footDispWrap = fel; 
47200         this.footDispWrap.overflow  = 'hidden';
47201         
47202         this.footDisp = fel.createChild();
47203         this.footDispWrap.on('click', this.onContextClick, this)
47204         
47205         
47206     },
47207     onContextClick : function (ev,dom)
47208     {
47209         ev.preventDefault();
47210         var  cn = dom.className;
47211         //Roo.log(cn);
47212         if (!cn.match(/x-ed-loc-/)) {
47213             return;
47214         }
47215         var n = cn.split('-').pop();
47216         var ans = this.footerEls;
47217         var sel = ans[n];
47218         
47219          // pick
47220         var range = this.editorcore.createRange();
47221         
47222         range.selectNodeContents(sel);
47223         //range.selectNode(sel);
47224         
47225         
47226         var selection = this.editorcore.getSelection();
47227         selection.removeAllRanges();
47228         selection.addRange(range);
47229         
47230         
47231         
47232         this.updateToolbar(null, null, sel);
47233         
47234         
47235     }
47236     
47237     
47238     
47239     
47240     
47241 });
47242
47243
47244
47245
47246
47247 /*
47248  * Based on:
47249  * Ext JS Library 1.1.1
47250  * Copyright(c) 2006-2007, Ext JS, LLC.
47251  *
47252  * Originally Released Under LGPL - original licence link has changed is not relivant.
47253  *
47254  * Fork - LGPL
47255  * <script type="text/javascript">
47256  */
47257  
47258 /**
47259  * @class Roo.form.BasicForm
47260  * @extends Roo.util.Observable
47261  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47262  * @constructor
47263  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47264  * @param {Object} config Configuration options
47265  */
47266 Roo.form.BasicForm = function(el, config){
47267     this.allItems = [];
47268     this.childForms = [];
47269     Roo.apply(this, config);
47270     /*
47271      * The Roo.form.Field items in this form.
47272      * @type MixedCollection
47273      */
47274      
47275      
47276     this.items = new Roo.util.MixedCollection(false, function(o){
47277         return o.id || (o.id = Roo.id());
47278     });
47279     this.addEvents({
47280         /**
47281          * @event beforeaction
47282          * Fires before any action is performed. Return false to cancel the action.
47283          * @param {Form} this
47284          * @param {Action} action The action to be performed
47285          */
47286         beforeaction: true,
47287         /**
47288          * @event actionfailed
47289          * Fires when an action fails.
47290          * @param {Form} this
47291          * @param {Action} action The action that failed
47292          */
47293         actionfailed : true,
47294         /**
47295          * @event actioncomplete
47296          * Fires when an action is completed.
47297          * @param {Form} this
47298          * @param {Action} action The action that completed
47299          */
47300         actioncomplete : true
47301     });
47302     if(el){
47303         this.initEl(el);
47304     }
47305     Roo.form.BasicForm.superclass.constructor.call(this);
47306     
47307     Roo.form.BasicForm.popover.apply();
47308 };
47309
47310 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47311     /**
47312      * @cfg {String} method
47313      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47314      */
47315     /**
47316      * @cfg {DataReader} reader
47317      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47318      * This is optional as there is built-in support for processing JSON.
47319      */
47320     /**
47321      * @cfg {DataReader} errorReader
47322      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47323      * This is completely optional as there is built-in support for processing JSON.
47324      */
47325     /**
47326      * @cfg {String} url
47327      * The URL to use for form actions if one isn't supplied in the action options.
47328      */
47329     /**
47330      * @cfg {Boolean} fileUpload
47331      * Set to true if this form is a file upload.
47332      */
47333      
47334     /**
47335      * @cfg {Object} baseParams
47336      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47337      */
47338      /**
47339      
47340     /**
47341      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47342      */
47343     timeout: 30,
47344
47345     // private
47346     activeAction : null,
47347
47348     /**
47349      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47350      * or setValues() data instead of when the form was first created.
47351      */
47352     trackResetOnLoad : false,
47353     
47354     
47355     /**
47356      * childForms - used for multi-tab forms
47357      * @type {Array}
47358      */
47359     childForms : false,
47360     
47361     /**
47362      * allItems - full list of fields.
47363      * @type {Array}
47364      */
47365     allItems : false,
47366     
47367     /**
47368      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47369      * element by passing it or its id or mask the form itself by passing in true.
47370      * @type Mixed
47371      */
47372     waitMsgTarget : false,
47373     
47374     /**
47375      * @type Boolean
47376      */
47377     disableMask : false,
47378     
47379     /**
47380      * @cfg {Boolean} errorMask (true|false) default false
47381      */
47382     errorMask : false,
47383     
47384     /**
47385      * @cfg {Number} maskOffset Default 100
47386      */
47387     maskOffset : 100,
47388
47389     // private
47390     initEl : function(el){
47391         this.el = Roo.get(el);
47392         this.id = this.el.id || Roo.id();
47393         this.el.on('submit', this.onSubmit, this);
47394         this.el.addClass('x-form');
47395     },
47396
47397     // private
47398     onSubmit : function(e){
47399         e.stopEvent();
47400     },
47401
47402     /**
47403      * Returns true if client-side validation on the form is successful.
47404      * @return Boolean
47405      */
47406     isValid : function(){
47407         var valid = true;
47408         var target = false;
47409         this.items.each(function(f){
47410             if(f.validate()){
47411                 return;
47412             }
47413             
47414             valid = false;
47415                 
47416             if(!target && f.el.isVisible(true)){
47417                 target = f;
47418             }
47419         });
47420         
47421         if(this.errorMask && !valid){
47422             Roo.form.BasicForm.popover.mask(this, target);
47423         }
47424         
47425         return valid;
47426     },
47427
47428     /**
47429      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47430      * @return Boolean
47431      */
47432     isDirty : function(){
47433         var dirty = false;
47434         this.items.each(function(f){
47435            if(f.isDirty()){
47436                dirty = true;
47437                return false;
47438            }
47439         });
47440         return dirty;
47441     },
47442     
47443     /**
47444      * Returns true if any fields in this form have changed since their original load. (New version)
47445      * @return Boolean
47446      */
47447     
47448     hasChanged : function()
47449     {
47450         var dirty = false;
47451         this.items.each(function(f){
47452            if(f.hasChanged()){
47453                dirty = true;
47454                return false;
47455            }
47456         });
47457         return dirty;
47458         
47459     },
47460     /**
47461      * Resets all hasChanged to 'false' -
47462      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47463      * So hasChanged storage is only to be used for this purpose
47464      * @return Boolean
47465      */
47466     resetHasChanged : function()
47467     {
47468         this.items.each(function(f){
47469            f.resetHasChanged();
47470         });
47471         
47472     },
47473     
47474     
47475     /**
47476      * Performs a predefined action (submit or load) or custom actions you define on this form.
47477      * @param {String} actionName The name of the action type
47478      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47479      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47480      * accept other config options):
47481      * <pre>
47482 Property          Type             Description
47483 ----------------  ---------------  ----------------------------------------------------------------------------------
47484 url               String           The url for the action (defaults to the form's url)
47485 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47486 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47487 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47488                                    validate the form on the client (defaults to false)
47489      * </pre>
47490      * @return {BasicForm} this
47491      */
47492     doAction : function(action, options){
47493         if(typeof action == 'string'){
47494             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47495         }
47496         if(this.fireEvent('beforeaction', this, action) !== false){
47497             this.beforeAction(action);
47498             action.run.defer(100, action);
47499         }
47500         return this;
47501     },
47502
47503     /**
47504      * Shortcut to do a submit action.
47505      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47506      * @return {BasicForm} this
47507      */
47508     submit : function(options){
47509         this.doAction('submit', options);
47510         return this;
47511     },
47512
47513     /**
47514      * Shortcut to do a load action.
47515      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47516      * @return {BasicForm} this
47517      */
47518     load : function(options){
47519         this.doAction('load', options);
47520         return this;
47521     },
47522
47523     /**
47524      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47525      * @param {Record} record The record to edit
47526      * @return {BasicForm} this
47527      */
47528     updateRecord : function(record){
47529         record.beginEdit();
47530         var fs = record.fields;
47531         fs.each(function(f){
47532             var field = this.findField(f.name);
47533             if(field){
47534                 record.set(f.name, field.getValue());
47535             }
47536         }, this);
47537         record.endEdit();
47538         return this;
47539     },
47540
47541     /**
47542      * Loads an Roo.data.Record into this form.
47543      * @param {Record} record The record to load
47544      * @return {BasicForm} this
47545      */
47546     loadRecord : function(record){
47547         this.setValues(record.data);
47548         return this;
47549     },
47550
47551     // private
47552     beforeAction : function(action){
47553         var o = action.options;
47554         
47555         if(!this.disableMask) {
47556             if(this.waitMsgTarget === true){
47557                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47558             }else if(this.waitMsgTarget){
47559                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47560                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47561             }else {
47562                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47563             }
47564         }
47565         
47566          
47567     },
47568
47569     // private
47570     afterAction : function(action, success){
47571         this.activeAction = null;
47572         var o = action.options;
47573         
47574         if(!this.disableMask) {
47575             if(this.waitMsgTarget === true){
47576                 this.el.unmask();
47577             }else if(this.waitMsgTarget){
47578                 this.waitMsgTarget.unmask();
47579             }else{
47580                 Roo.MessageBox.updateProgress(1);
47581                 Roo.MessageBox.hide();
47582             }
47583         }
47584         
47585         if(success){
47586             if(o.reset){
47587                 this.reset();
47588             }
47589             Roo.callback(o.success, o.scope, [this, action]);
47590             this.fireEvent('actioncomplete', this, action);
47591             
47592         }else{
47593             
47594             // failure condition..
47595             // we have a scenario where updates need confirming.
47596             // eg. if a locking scenario exists..
47597             // we look for { errors : { needs_confirm : true }} in the response.
47598             if (
47599                 (typeof(action.result) != 'undefined')  &&
47600                 (typeof(action.result.errors) != 'undefined')  &&
47601                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47602            ){
47603                 var _t = this;
47604                 Roo.MessageBox.confirm(
47605                     "Change requires confirmation",
47606                     action.result.errorMsg,
47607                     function(r) {
47608                         if (r != 'yes') {
47609                             return;
47610                         }
47611                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47612                     }
47613                     
47614                 );
47615                 
47616                 
47617                 
47618                 return;
47619             }
47620             
47621             Roo.callback(o.failure, o.scope, [this, action]);
47622             // show an error message if no failed handler is set..
47623             if (!this.hasListener('actionfailed')) {
47624                 Roo.MessageBox.alert("Error",
47625                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47626                         action.result.errorMsg :
47627                         "Saving Failed, please check your entries or try again"
47628                 );
47629             }
47630             
47631             this.fireEvent('actionfailed', this, action);
47632         }
47633         
47634     },
47635
47636     /**
47637      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47638      * @param {String} id The value to search for
47639      * @return Field
47640      */
47641     findField : function(id){
47642         var field = this.items.get(id);
47643         if(!field){
47644             this.items.each(function(f){
47645                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47646                     field = f;
47647                     return false;
47648                 }
47649             });
47650         }
47651         return field || null;
47652     },
47653
47654     /**
47655      * Add a secondary form to this one, 
47656      * Used to provide tabbed forms. One form is primary, with hidden values 
47657      * which mirror the elements from the other forms.
47658      * 
47659      * @param {Roo.form.Form} form to add.
47660      * 
47661      */
47662     addForm : function(form)
47663     {
47664        
47665         if (this.childForms.indexOf(form) > -1) {
47666             // already added..
47667             return;
47668         }
47669         this.childForms.push(form);
47670         var n = '';
47671         Roo.each(form.allItems, function (fe) {
47672             
47673             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47674             if (this.findField(n)) { // already added..
47675                 return;
47676             }
47677             var add = new Roo.form.Hidden({
47678                 name : n
47679             });
47680             add.render(this.el);
47681             
47682             this.add( add );
47683         }, this);
47684         
47685     },
47686     /**
47687      * Mark fields in this form invalid in bulk.
47688      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47689      * @return {BasicForm} this
47690      */
47691     markInvalid : function(errors){
47692         if(errors instanceof Array){
47693             for(var i = 0, len = errors.length; i < len; i++){
47694                 var fieldError = errors[i];
47695                 var f = this.findField(fieldError.id);
47696                 if(f){
47697                     f.markInvalid(fieldError.msg);
47698                 }
47699             }
47700         }else{
47701             var field, id;
47702             for(id in errors){
47703                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47704                     field.markInvalid(errors[id]);
47705                 }
47706             }
47707         }
47708         Roo.each(this.childForms || [], function (f) {
47709             f.markInvalid(errors);
47710         });
47711         
47712         return this;
47713     },
47714
47715     /**
47716      * Set values for fields in this form in bulk.
47717      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47718      * @return {BasicForm} this
47719      */
47720     setValues : function(values){
47721         if(values instanceof Array){ // array of objects
47722             for(var i = 0, len = values.length; i < len; i++){
47723                 var v = values[i];
47724                 var f = this.findField(v.id);
47725                 if(f){
47726                     f.setValue(v.value);
47727                     if(this.trackResetOnLoad){
47728                         f.originalValue = f.getValue();
47729                     }
47730                 }
47731             }
47732         }else{ // object hash
47733             var field, id;
47734             for(id in values){
47735                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47736                     
47737                     if (field.setFromData && 
47738                         field.valueField && 
47739                         field.displayField &&
47740                         // combos' with local stores can 
47741                         // be queried via setValue()
47742                         // to set their value..
47743                         (field.store && !field.store.isLocal)
47744                         ) {
47745                         // it's a combo
47746                         var sd = { };
47747                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47748                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47749                         field.setFromData(sd);
47750                         
47751                     } else {
47752                         field.setValue(values[id]);
47753                     }
47754                     
47755                     
47756                     if(this.trackResetOnLoad){
47757                         field.originalValue = field.getValue();
47758                     }
47759                 }
47760             }
47761         }
47762         this.resetHasChanged();
47763         
47764         
47765         Roo.each(this.childForms || [], function (f) {
47766             f.setValues(values);
47767             f.resetHasChanged();
47768         });
47769                 
47770         return this;
47771     },
47772  
47773     /**
47774      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47775      * they are returned as an array.
47776      * @param {Boolean} asString
47777      * @return {Object}
47778      */
47779     getValues : function(asString){
47780         if (this.childForms) {
47781             // copy values from the child forms
47782             Roo.each(this.childForms, function (f) {
47783                 this.setValues(f.getValues());
47784             }, this);
47785         }
47786         
47787         // use formdata
47788         if (typeof(FormData) != 'undefined' && asString !== true) {
47789             var fd = (new FormData(this.el.dom)).entries();
47790             var ret = {};
47791             var ent = fd.next();
47792             while (!ent.done) {
47793                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47794                 ent = fd.next();
47795             };
47796             return ret;
47797         }
47798         
47799         
47800         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47801         if(asString === true){
47802             return fs;
47803         }
47804         return Roo.urlDecode(fs);
47805     },
47806     
47807     /**
47808      * Returns the fields in this form as an object with key/value pairs. 
47809      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47810      * @return {Object}
47811      */
47812     getFieldValues : function(with_hidden)
47813     {
47814         if (this.childForms) {
47815             // copy values from the child forms
47816             // should this call getFieldValues - probably not as we do not currently copy
47817             // hidden fields when we generate..
47818             Roo.each(this.childForms, function (f) {
47819                 this.setValues(f.getValues());
47820             }, this);
47821         }
47822         
47823         var ret = {};
47824         this.items.each(function(f){
47825             if (!f.getName()) {
47826                 return;
47827             }
47828             var v = f.getValue();
47829             if (f.inputType =='radio') {
47830                 if (typeof(ret[f.getName()]) == 'undefined') {
47831                     ret[f.getName()] = ''; // empty..
47832                 }
47833                 
47834                 if (!f.el.dom.checked) {
47835                     return;
47836                     
47837                 }
47838                 v = f.el.dom.value;
47839                 
47840             }
47841             
47842             // not sure if this supported any more..
47843             if ((typeof(v) == 'object') && f.getRawValue) {
47844                 v = f.getRawValue() ; // dates..
47845             }
47846             // combo boxes where name != hiddenName...
47847             if (f.name != f.getName()) {
47848                 ret[f.name] = f.getRawValue();
47849             }
47850             ret[f.getName()] = v;
47851         });
47852         
47853         return ret;
47854     },
47855
47856     /**
47857      * Clears all invalid messages in this form.
47858      * @return {BasicForm} this
47859      */
47860     clearInvalid : function(){
47861         this.items.each(function(f){
47862            f.clearInvalid();
47863         });
47864         
47865         Roo.each(this.childForms || [], function (f) {
47866             f.clearInvalid();
47867         });
47868         
47869         
47870         return this;
47871     },
47872
47873     /**
47874      * Resets this form.
47875      * @return {BasicForm} this
47876      */
47877     reset : function(){
47878         this.items.each(function(f){
47879             f.reset();
47880         });
47881         
47882         Roo.each(this.childForms || [], function (f) {
47883             f.reset();
47884         });
47885         this.resetHasChanged();
47886         
47887         return this;
47888     },
47889
47890     /**
47891      * Add Roo.form components to this form.
47892      * @param {Field} field1
47893      * @param {Field} field2 (optional)
47894      * @param {Field} etc (optional)
47895      * @return {BasicForm} this
47896      */
47897     add : function(){
47898         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47899         return this;
47900     },
47901
47902
47903     /**
47904      * Removes a field from the items collection (does NOT remove its markup).
47905      * @param {Field} field
47906      * @return {BasicForm} this
47907      */
47908     remove : function(field){
47909         this.items.remove(field);
47910         return this;
47911     },
47912
47913     /**
47914      * Looks at the fields in this form, checks them for an id attribute,
47915      * and calls applyTo on the existing dom element with that id.
47916      * @return {BasicForm} this
47917      */
47918     render : function(){
47919         this.items.each(function(f){
47920             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47921                 f.applyTo(f.id);
47922             }
47923         });
47924         return this;
47925     },
47926
47927     /**
47928      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47929      * @param {Object} values
47930      * @return {BasicForm} this
47931      */
47932     applyToFields : function(o){
47933         this.items.each(function(f){
47934            Roo.apply(f, o);
47935         });
47936         return this;
47937     },
47938
47939     /**
47940      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47941      * @param {Object} values
47942      * @return {BasicForm} this
47943      */
47944     applyIfToFields : function(o){
47945         this.items.each(function(f){
47946            Roo.applyIf(f, o);
47947         });
47948         return this;
47949     }
47950 });
47951
47952 // back compat
47953 Roo.BasicForm = Roo.form.BasicForm;
47954
47955 Roo.apply(Roo.form.BasicForm, {
47956     
47957     popover : {
47958         
47959         padding : 5,
47960         
47961         isApplied : false,
47962         
47963         isMasked : false,
47964         
47965         form : false,
47966         
47967         target : false,
47968         
47969         intervalID : false,
47970         
47971         maskEl : false,
47972         
47973         apply : function()
47974         {
47975             if(this.isApplied){
47976                 return;
47977             }
47978             
47979             this.maskEl = {
47980                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
47981                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
47982                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
47983                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
47984             };
47985             
47986             this.maskEl.top.enableDisplayMode("block");
47987             this.maskEl.left.enableDisplayMode("block");
47988             this.maskEl.bottom.enableDisplayMode("block");
47989             this.maskEl.right.enableDisplayMode("block");
47990             
47991             Roo.get(document.body).on('click', function(){
47992                 this.unmask();
47993             }, this);
47994             
47995             Roo.get(document.body).on('touchstart', function(){
47996                 this.unmask();
47997             }, this);
47998             
47999             this.isApplied = true
48000         },
48001         
48002         mask : function(form, target)
48003         {
48004             this.form = form;
48005             
48006             this.target = target;
48007             
48008             if(!this.form.errorMask || !target.el){
48009                 return;
48010             }
48011             
48012             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
48013             
48014             var ot = this.target.el.calcOffsetsTo(scrollable);
48015             
48016             var scrollTo = ot[1] - this.form.maskOffset;
48017             
48018             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48019             
48020             scrollable.scrollTo('top', scrollTo);
48021             
48022             var el = this.target.wrap || this.target.el;
48023             
48024             var box = el.getBox();
48025             
48026             this.maskEl.top.setStyle('position', 'absolute');
48027             this.maskEl.top.setStyle('z-index', 10000);
48028             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48029             this.maskEl.top.setLeft(0);
48030             this.maskEl.top.setTop(0);
48031             this.maskEl.top.show();
48032             
48033             this.maskEl.left.setStyle('position', 'absolute');
48034             this.maskEl.left.setStyle('z-index', 10000);
48035             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48036             this.maskEl.left.setLeft(0);
48037             this.maskEl.left.setTop(box.y - this.padding);
48038             this.maskEl.left.show();
48039
48040             this.maskEl.bottom.setStyle('position', 'absolute');
48041             this.maskEl.bottom.setStyle('z-index', 10000);
48042             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48043             this.maskEl.bottom.setLeft(0);
48044             this.maskEl.bottom.setTop(box.bottom + this.padding);
48045             this.maskEl.bottom.show();
48046
48047             this.maskEl.right.setStyle('position', 'absolute');
48048             this.maskEl.right.setStyle('z-index', 10000);
48049             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48050             this.maskEl.right.setLeft(box.right + this.padding);
48051             this.maskEl.right.setTop(box.y - this.padding);
48052             this.maskEl.right.show();
48053
48054             this.intervalID = window.setInterval(function() {
48055                 Roo.form.BasicForm.popover.unmask();
48056             }, 10000);
48057
48058             window.onwheel = function(){ return false;};
48059             
48060             (function(){ this.isMasked = true; }).defer(500, this);
48061             
48062         },
48063         
48064         unmask : function()
48065         {
48066             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48067                 return;
48068             }
48069             
48070             this.maskEl.top.setStyle('position', 'absolute');
48071             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48072             this.maskEl.top.hide();
48073
48074             this.maskEl.left.setStyle('position', 'absolute');
48075             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48076             this.maskEl.left.hide();
48077
48078             this.maskEl.bottom.setStyle('position', 'absolute');
48079             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48080             this.maskEl.bottom.hide();
48081
48082             this.maskEl.right.setStyle('position', 'absolute');
48083             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48084             this.maskEl.right.hide();
48085             
48086             window.onwheel = function(){ return true;};
48087             
48088             if(this.intervalID){
48089                 window.clearInterval(this.intervalID);
48090                 this.intervalID = false;
48091             }
48092             
48093             this.isMasked = false;
48094             
48095         }
48096         
48097     }
48098     
48099 });/*
48100  * Based on:
48101  * Ext JS Library 1.1.1
48102  * Copyright(c) 2006-2007, Ext JS, LLC.
48103  *
48104  * Originally Released Under LGPL - original licence link has changed is not relivant.
48105  *
48106  * Fork - LGPL
48107  * <script type="text/javascript">
48108  */
48109
48110 /**
48111  * @class Roo.form.Form
48112  * @extends Roo.form.BasicForm
48113  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48114  * @constructor
48115  * @param {Object} config Configuration options
48116  */
48117 Roo.form.Form = function(config){
48118     var xitems =  [];
48119     if (config.items) {
48120         xitems = config.items;
48121         delete config.items;
48122     }
48123    
48124     
48125     Roo.form.Form.superclass.constructor.call(this, null, config);
48126     this.url = this.url || this.action;
48127     if(!this.root){
48128         this.root = new Roo.form.Layout(Roo.applyIf({
48129             id: Roo.id()
48130         }, config));
48131     }
48132     this.active = this.root;
48133     /**
48134      * Array of all the buttons that have been added to this form via {@link addButton}
48135      * @type Array
48136      */
48137     this.buttons = [];
48138     this.allItems = [];
48139     this.addEvents({
48140         /**
48141          * @event clientvalidation
48142          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48143          * @param {Form} this
48144          * @param {Boolean} valid true if the form has passed client-side validation
48145          */
48146         clientvalidation: true,
48147         /**
48148          * @event rendered
48149          * Fires when the form is rendered
48150          * @param {Roo.form.Form} form
48151          */
48152         rendered : true
48153     });
48154     
48155     if (this.progressUrl) {
48156             // push a hidden field onto the list of fields..
48157             this.addxtype( {
48158                     xns: Roo.form, 
48159                     xtype : 'Hidden', 
48160                     name : 'UPLOAD_IDENTIFIER' 
48161             });
48162         }
48163         
48164     
48165     Roo.each(xitems, this.addxtype, this);
48166     
48167 };
48168
48169 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48170     /**
48171      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48172      */
48173     /**
48174      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48175      */
48176     /**
48177      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48178      */
48179     buttonAlign:'center',
48180
48181     /**
48182      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48183      */
48184     minButtonWidth:75,
48185
48186     /**
48187      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48188      * This property cascades to child containers if not set.
48189      */
48190     labelAlign:'left',
48191
48192     /**
48193      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48194      * fires a looping event with that state. This is required to bind buttons to the valid
48195      * state using the config value formBind:true on the button.
48196      */
48197     monitorValid : false,
48198
48199     /**
48200      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48201      */
48202     monitorPoll : 200,
48203     
48204     /**
48205      * @cfg {String} progressUrl - Url to return progress data 
48206      */
48207     
48208     progressUrl : false,
48209     /**
48210      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48211      * sending a formdata with extra parameters - eg uploaded elements.
48212      */
48213     
48214     formData : false,
48215     
48216     /**
48217      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48218      * fields are added and the column is closed. If no fields are passed the column remains open
48219      * until end() is called.
48220      * @param {Object} config The config to pass to the column
48221      * @param {Field} field1 (optional)
48222      * @param {Field} field2 (optional)
48223      * @param {Field} etc (optional)
48224      * @return Column The column container object
48225      */
48226     column : function(c){
48227         var col = new Roo.form.Column(c);
48228         this.start(col);
48229         if(arguments.length > 1){ // duplicate code required because of Opera
48230             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48231             this.end();
48232         }
48233         return col;
48234     },
48235
48236     /**
48237      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48238      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48239      * until end() is called.
48240      * @param {Object} config The config to pass to the fieldset
48241      * @param {Field} field1 (optional)
48242      * @param {Field} field2 (optional)
48243      * @param {Field} etc (optional)
48244      * @return FieldSet The fieldset container object
48245      */
48246     fieldset : function(c){
48247         var fs = new Roo.form.FieldSet(c);
48248         this.start(fs);
48249         if(arguments.length > 1){ // duplicate code required because of Opera
48250             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48251             this.end();
48252         }
48253         return fs;
48254     },
48255
48256     /**
48257      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48258      * fields are added and the container is closed. If no fields are passed the container remains open
48259      * until end() is called.
48260      * @param {Object} config The config to pass to the Layout
48261      * @param {Field} field1 (optional)
48262      * @param {Field} field2 (optional)
48263      * @param {Field} etc (optional)
48264      * @return Layout The container object
48265      */
48266     container : function(c){
48267         var l = new Roo.form.Layout(c);
48268         this.start(l);
48269         if(arguments.length > 1){ // duplicate code required because of Opera
48270             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48271             this.end();
48272         }
48273         return l;
48274     },
48275
48276     /**
48277      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48278      * @param {Object} container A Roo.form.Layout or subclass of Layout
48279      * @return {Form} this
48280      */
48281     start : function(c){
48282         // cascade label info
48283         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48284         this.active.stack.push(c);
48285         c.ownerCt = this.active;
48286         this.active = c;
48287         return this;
48288     },
48289
48290     /**
48291      * Closes the current open container
48292      * @return {Form} this
48293      */
48294     end : function(){
48295         if(this.active == this.root){
48296             return this;
48297         }
48298         this.active = this.active.ownerCt;
48299         return this;
48300     },
48301
48302     /**
48303      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48304      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48305      * as the label of the field.
48306      * @param {Field} field1
48307      * @param {Field} field2 (optional)
48308      * @param {Field} etc. (optional)
48309      * @return {Form} this
48310      */
48311     add : function(){
48312         this.active.stack.push.apply(this.active.stack, arguments);
48313         this.allItems.push.apply(this.allItems,arguments);
48314         var r = [];
48315         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48316             if(a[i].isFormField){
48317                 r.push(a[i]);
48318             }
48319         }
48320         if(r.length > 0){
48321             Roo.form.Form.superclass.add.apply(this, r);
48322         }
48323         return this;
48324     },
48325     
48326
48327     
48328     
48329     
48330      /**
48331      * Find any element that has been added to a form, using it's ID or name
48332      * This can include framesets, columns etc. along with regular fields..
48333      * @param {String} id - id or name to find.
48334      
48335      * @return {Element} e - or false if nothing found.
48336      */
48337     findbyId : function(id)
48338     {
48339         var ret = false;
48340         if (!id) {
48341             return ret;
48342         }
48343         Roo.each(this.allItems, function(f){
48344             if (f.id == id || f.name == id ){
48345                 ret = f;
48346                 return false;
48347             }
48348         });
48349         return ret;
48350     },
48351
48352     
48353     
48354     /**
48355      * Render this form into the passed container. This should only be called once!
48356      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48357      * @return {Form} this
48358      */
48359     render : function(ct)
48360     {
48361         
48362         
48363         
48364         ct = Roo.get(ct);
48365         var o = this.autoCreate || {
48366             tag: 'form',
48367             method : this.method || 'POST',
48368             id : this.id || Roo.id()
48369         };
48370         this.initEl(ct.createChild(o));
48371
48372         this.root.render(this.el);
48373         
48374        
48375              
48376         this.items.each(function(f){
48377             f.render('x-form-el-'+f.id);
48378         });
48379
48380         if(this.buttons.length > 0){
48381             // tables are required to maintain order and for correct IE layout
48382             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48383                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48384                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48385             }}, null, true);
48386             var tr = tb.getElementsByTagName('tr')[0];
48387             for(var i = 0, len = this.buttons.length; i < len; i++) {
48388                 var b = this.buttons[i];
48389                 var td = document.createElement('td');
48390                 td.className = 'x-form-btn-td';
48391                 b.render(tr.appendChild(td));
48392             }
48393         }
48394         if(this.monitorValid){ // initialize after render
48395             this.startMonitoring();
48396         }
48397         this.fireEvent('rendered', this);
48398         return this;
48399     },
48400
48401     /**
48402      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48403      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48404      * object or a valid Roo.DomHelper element config
48405      * @param {Function} handler The function called when the button is clicked
48406      * @param {Object} scope (optional) The scope of the handler function
48407      * @return {Roo.Button}
48408      */
48409     addButton : function(config, handler, scope){
48410         var bc = {
48411             handler: handler,
48412             scope: scope,
48413             minWidth: this.minButtonWidth,
48414             hideParent:true
48415         };
48416         if(typeof config == "string"){
48417             bc.text = config;
48418         }else{
48419             Roo.apply(bc, config);
48420         }
48421         var btn = new Roo.Button(null, bc);
48422         this.buttons.push(btn);
48423         return btn;
48424     },
48425
48426      /**
48427      * Adds a series of form elements (using the xtype property as the factory method.
48428      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48429      * @param {Object} config 
48430      */
48431     
48432     addxtype : function()
48433     {
48434         var ar = Array.prototype.slice.call(arguments, 0);
48435         var ret = false;
48436         for(var i = 0; i < ar.length; i++) {
48437             if (!ar[i]) {
48438                 continue; // skip -- if this happends something invalid got sent, we 
48439                 // should ignore it, as basically that interface element will not show up
48440                 // and that should be pretty obvious!!
48441             }
48442             
48443             if (Roo.form[ar[i].xtype]) {
48444                 ar[i].form = this;
48445                 var fe = Roo.factory(ar[i], Roo.form);
48446                 if (!ret) {
48447                     ret = fe;
48448                 }
48449                 fe.form = this;
48450                 if (fe.store) {
48451                     fe.store.form = this;
48452                 }
48453                 if (fe.isLayout) {  
48454                          
48455                     this.start(fe);
48456                     this.allItems.push(fe);
48457                     if (fe.items && fe.addxtype) {
48458                         fe.addxtype.apply(fe, fe.items);
48459                         delete fe.items;
48460                     }
48461                      this.end();
48462                     continue;
48463                 }
48464                 
48465                 
48466                  
48467                 this.add(fe);
48468               //  console.log('adding ' + ar[i].xtype);
48469             }
48470             if (ar[i].xtype == 'Button') {  
48471                 //console.log('adding button');
48472                 //console.log(ar[i]);
48473                 this.addButton(ar[i]);
48474                 this.allItems.push(fe);
48475                 continue;
48476             }
48477             
48478             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48479                 alert('end is not supported on xtype any more, use items');
48480             //    this.end();
48481             //    //console.log('adding end');
48482             }
48483             
48484         }
48485         return ret;
48486     },
48487     
48488     /**
48489      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48490      * option "monitorValid"
48491      */
48492     startMonitoring : function(){
48493         if(!this.bound){
48494             this.bound = true;
48495             Roo.TaskMgr.start({
48496                 run : this.bindHandler,
48497                 interval : this.monitorPoll || 200,
48498                 scope: this
48499             });
48500         }
48501     },
48502
48503     /**
48504      * Stops monitoring of the valid state of this form
48505      */
48506     stopMonitoring : function(){
48507         this.bound = false;
48508     },
48509
48510     // private
48511     bindHandler : function(){
48512         if(!this.bound){
48513             return false; // stops binding
48514         }
48515         var valid = true;
48516         this.items.each(function(f){
48517             if(!f.isValid(true)){
48518                 valid = false;
48519                 return false;
48520             }
48521         });
48522         for(var i = 0, len = this.buttons.length; i < len; i++){
48523             var btn = this.buttons[i];
48524             if(btn.formBind === true && btn.disabled === valid){
48525                 btn.setDisabled(!valid);
48526             }
48527         }
48528         this.fireEvent('clientvalidation', this, valid);
48529     }
48530     
48531     
48532     
48533     
48534     
48535     
48536     
48537     
48538 });
48539
48540
48541 // back compat
48542 Roo.Form = Roo.form.Form;
48543 /*
48544  * Based on:
48545  * Ext JS Library 1.1.1
48546  * Copyright(c) 2006-2007, Ext JS, LLC.
48547  *
48548  * Originally Released Under LGPL - original licence link has changed is not relivant.
48549  *
48550  * Fork - LGPL
48551  * <script type="text/javascript">
48552  */
48553
48554 // as we use this in bootstrap.
48555 Roo.namespace('Roo.form');
48556  /**
48557  * @class Roo.form.Action
48558  * Internal Class used to handle form actions
48559  * @constructor
48560  * @param {Roo.form.BasicForm} el The form element or its id
48561  * @param {Object} config Configuration options
48562  */
48563
48564  
48565  
48566 // define the action interface
48567 Roo.form.Action = function(form, options){
48568     this.form = form;
48569     this.options = options || {};
48570 };
48571 /**
48572  * Client Validation Failed
48573  * @const 
48574  */
48575 Roo.form.Action.CLIENT_INVALID = 'client';
48576 /**
48577  * Server Validation Failed
48578  * @const 
48579  */
48580 Roo.form.Action.SERVER_INVALID = 'server';
48581  /**
48582  * Connect to Server Failed
48583  * @const 
48584  */
48585 Roo.form.Action.CONNECT_FAILURE = 'connect';
48586 /**
48587  * Reading Data from Server Failed
48588  * @const 
48589  */
48590 Roo.form.Action.LOAD_FAILURE = 'load';
48591
48592 Roo.form.Action.prototype = {
48593     type : 'default',
48594     failureType : undefined,
48595     response : undefined,
48596     result : undefined,
48597
48598     // interface method
48599     run : function(options){
48600
48601     },
48602
48603     // interface method
48604     success : function(response){
48605
48606     },
48607
48608     // interface method
48609     handleResponse : function(response){
48610
48611     },
48612
48613     // default connection failure
48614     failure : function(response){
48615         
48616         this.response = response;
48617         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48618         this.form.afterAction(this, false);
48619     },
48620
48621     processResponse : function(response){
48622         this.response = response;
48623         if(!response.responseText){
48624             return true;
48625         }
48626         this.result = this.handleResponse(response);
48627         return this.result;
48628     },
48629
48630     // utility functions used internally
48631     getUrl : function(appendParams){
48632         var url = this.options.url || this.form.url || this.form.el.dom.action;
48633         if(appendParams){
48634             var p = this.getParams();
48635             if(p){
48636                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48637             }
48638         }
48639         return url;
48640     },
48641
48642     getMethod : function(){
48643         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48644     },
48645
48646     getParams : function(){
48647         var bp = this.form.baseParams;
48648         var p = this.options.params;
48649         if(p){
48650             if(typeof p == "object"){
48651                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48652             }else if(typeof p == 'string' && bp){
48653                 p += '&' + Roo.urlEncode(bp);
48654             }
48655         }else if(bp){
48656             p = Roo.urlEncode(bp);
48657         }
48658         return p;
48659     },
48660
48661     createCallback : function(){
48662         return {
48663             success: this.success,
48664             failure: this.failure,
48665             scope: this,
48666             timeout: (this.form.timeout*1000),
48667             upload: this.form.fileUpload ? this.success : undefined
48668         };
48669     }
48670 };
48671
48672 Roo.form.Action.Submit = function(form, options){
48673     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48674 };
48675
48676 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48677     type : 'submit',
48678
48679     haveProgress : false,
48680     uploadComplete : false,
48681     
48682     // uploadProgress indicator.
48683     uploadProgress : function()
48684     {
48685         if (!this.form.progressUrl) {
48686             return;
48687         }
48688         
48689         if (!this.haveProgress) {
48690             Roo.MessageBox.progress("Uploading", "Uploading");
48691         }
48692         if (this.uploadComplete) {
48693            Roo.MessageBox.hide();
48694            return;
48695         }
48696         
48697         this.haveProgress = true;
48698    
48699         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48700         
48701         var c = new Roo.data.Connection();
48702         c.request({
48703             url : this.form.progressUrl,
48704             params: {
48705                 id : uid
48706             },
48707             method: 'GET',
48708             success : function(req){
48709                //console.log(data);
48710                 var rdata = false;
48711                 var edata;
48712                 try  {
48713                    rdata = Roo.decode(req.responseText)
48714                 } catch (e) {
48715                     Roo.log("Invalid data from server..");
48716                     Roo.log(edata);
48717                     return;
48718                 }
48719                 if (!rdata || !rdata.success) {
48720                     Roo.log(rdata);
48721                     Roo.MessageBox.alert(Roo.encode(rdata));
48722                     return;
48723                 }
48724                 var data = rdata.data;
48725                 
48726                 if (this.uploadComplete) {
48727                    Roo.MessageBox.hide();
48728                    return;
48729                 }
48730                    
48731                 if (data){
48732                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48733                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48734                     );
48735                 }
48736                 this.uploadProgress.defer(2000,this);
48737             },
48738        
48739             failure: function(data) {
48740                 Roo.log('progress url failed ');
48741                 Roo.log(data);
48742             },
48743             scope : this
48744         });
48745            
48746     },
48747     
48748     
48749     run : function()
48750     {
48751         // run get Values on the form, so it syncs any secondary forms.
48752         this.form.getValues();
48753         
48754         var o = this.options;
48755         var method = this.getMethod();
48756         var isPost = method == 'POST';
48757         if(o.clientValidation === false || this.form.isValid()){
48758             
48759             if (this.form.progressUrl) {
48760                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48761                     (new Date() * 1) + '' + Math.random());
48762                     
48763             } 
48764             
48765             
48766             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48767                 form:this.form.el.dom,
48768                 url:this.getUrl(!isPost),
48769                 method: method,
48770                 params:isPost ? this.getParams() : null,
48771                 isUpload: this.form.fileUpload,
48772                 formData : this.form.formData
48773             }));
48774             
48775             this.uploadProgress();
48776
48777         }else if (o.clientValidation !== false){ // client validation failed
48778             this.failureType = Roo.form.Action.CLIENT_INVALID;
48779             this.form.afterAction(this, false);
48780         }
48781     },
48782
48783     success : function(response)
48784     {
48785         this.uploadComplete= true;
48786         if (this.haveProgress) {
48787             Roo.MessageBox.hide();
48788         }
48789         
48790         
48791         var result = this.processResponse(response);
48792         if(result === true || result.success){
48793             this.form.afterAction(this, true);
48794             return;
48795         }
48796         if(result.errors){
48797             this.form.markInvalid(result.errors);
48798             this.failureType = Roo.form.Action.SERVER_INVALID;
48799         }
48800         this.form.afterAction(this, false);
48801     },
48802     failure : function(response)
48803     {
48804         this.uploadComplete= true;
48805         if (this.haveProgress) {
48806             Roo.MessageBox.hide();
48807         }
48808         
48809         this.response = response;
48810         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48811         this.form.afterAction(this, false);
48812     },
48813     
48814     handleResponse : function(response){
48815         if(this.form.errorReader){
48816             var rs = this.form.errorReader.read(response);
48817             var errors = [];
48818             if(rs.records){
48819                 for(var i = 0, len = rs.records.length; i < len; i++) {
48820                     var r = rs.records[i];
48821                     errors[i] = r.data;
48822                 }
48823             }
48824             if(errors.length < 1){
48825                 errors = null;
48826             }
48827             return {
48828                 success : rs.success,
48829                 errors : errors
48830             };
48831         }
48832         var ret = false;
48833         try {
48834             ret = Roo.decode(response.responseText);
48835         } catch (e) {
48836             ret = {
48837                 success: false,
48838                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48839                 errors : []
48840             };
48841         }
48842         return ret;
48843         
48844     }
48845 });
48846
48847
48848 Roo.form.Action.Load = function(form, options){
48849     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48850     this.reader = this.form.reader;
48851 };
48852
48853 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48854     type : 'load',
48855
48856     run : function(){
48857         
48858         Roo.Ajax.request(Roo.apply(
48859                 this.createCallback(), {
48860                     method:this.getMethod(),
48861                     url:this.getUrl(false),
48862                     params:this.getParams()
48863         }));
48864     },
48865
48866     success : function(response){
48867         
48868         var result = this.processResponse(response);
48869         if(result === true || !result.success || !result.data){
48870             this.failureType = Roo.form.Action.LOAD_FAILURE;
48871             this.form.afterAction(this, false);
48872             return;
48873         }
48874         this.form.clearInvalid();
48875         this.form.setValues(result.data);
48876         this.form.afterAction(this, true);
48877     },
48878
48879     handleResponse : function(response){
48880         if(this.form.reader){
48881             var rs = this.form.reader.read(response);
48882             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48883             return {
48884                 success : rs.success,
48885                 data : data
48886             };
48887         }
48888         return Roo.decode(response.responseText);
48889     }
48890 });
48891
48892 Roo.form.Action.ACTION_TYPES = {
48893     'load' : Roo.form.Action.Load,
48894     'submit' : Roo.form.Action.Submit
48895 };/*
48896  * Based on:
48897  * Ext JS Library 1.1.1
48898  * Copyright(c) 2006-2007, Ext JS, LLC.
48899  *
48900  * Originally Released Under LGPL - original licence link has changed is not relivant.
48901  *
48902  * Fork - LGPL
48903  * <script type="text/javascript">
48904  */
48905  
48906 /**
48907  * @class Roo.form.Layout
48908  * @extends Roo.Component
48909  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48910  * @constructor
48911  * @param {Object} config Configuration options
48912  */
48913 Roo.form.Layout = function(config){
48914     var xitems = [];
48915     if (config.items) {
48916         xitems = config.items;
48917         delete config.items;
48918     }
48919     Roo.form.Layout.superclass.constructor.call(this, config);
48920     this.stack = [];
48921     Roo.each(xitems, this.addxtype, this);
48922      
48923 };
48924
48925 Roo.extend(Roo.form.Layout, Roo.Component, {
48926     /**
48927      * @cfg {String/Object} autoCreate
48928      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48929      */
48930     /**
48931      * @cfg {String/Object/Function} style
48932      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48933      * a function which returns such a specification.
48934      */
48935     /**
48936      * @cfg {String} labelAlign
48937      * Valid values are "left," "top" and "right" (defaults to "left")
48938      */
48939     /**
48940      * @cfg {Number} labelWidth
48941      * Fixed width in pixels of all field labels (defaults to undefined)
48942      */
48943     /**
48944      * @cfg {Boolean} clear
48945      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48946      */
48947     clear : true,
48948     /**
48949      * @cfg {String} labelSeparator
48950      * The separator to use after field labels (defaults to ':')
48951      */
48952     labelSeparator : ':',
48953     /**
48954      * @cfg {Boolean} hideLabels
48955      * True to suppress the display of field labels in this layout (defaults to false)
48956      */
48957     hideLabels : false,
48958
48959     // private
48960     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48961     
48962     isLayout : true,
48963     
48964     // private
48965     onRender : function(ct, position){
48966         if(this.el){ // from markup
48967             this.el = Roo.get(this.el);
48968         }else {  // generate
48969             var cfg = this.getAutoCreate();
48970             this.el = ct.createChild(cfg, position);
48971         }
48972         if(this.style){
48973             this.el.applyStyles(this.style);
48974         }
48975         if(this.labelAlign){
48976             this.el.addClass('x-form-label-'+this.labelAlign);
48977         }
48978         if(this.hideLabels){
48979             this.labelStyle = "display:none";
48980             this.elementStyle = "padding-left:0;";
48981         }else{
48982             if(typeof this.labelWidth == 'number'){
48983                 this.labelStyle = "width:"+this.labelWidth+"px;";
48984                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48985             }
48986             if(this.labelAlign == 'top'){
48987                 this.labelStyle = "width:auto;";
48988                 this.elementStyle = "padding-left:0;";
48989             }
48990         }
48991         var stack = this.stack;
48992         var slen = stack.length;
48993         if(slen > 0){
48994             if(!this.fieldTpl){
48995                 var t = new Roo.Template(
48996                     '<div class="x-form-item {5}">',
48997                         '<label for="{0}" style="{2}">{1}{4}</label>',
48998                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48999                         '</div>',
49000                     '</div><div class="x-form-clear-left"></div>'
49001                 );
49002                 t.disableFormats = true;
49003                 t.compile();
49004                 Roo.form.Layout.prototype.fieldTpl = t;
49005             }
49006             for(var i = 0; i < slen; i++) {
49007                 if(stack[i].isFormField){
49008                     this.renderField(stack[i]);
49009                 }else{
49010                     this.renderComponent(stack[i]);
49011                 }
49012             }
49013         }
49014         if(this.clear){
49015             this.el.createChild({cls:'x-form-clear'});
49016         }
49017     },
49018
49019     // private
49020     renderField : function(f){
49021         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49022                f.id, //0
49023                f.fieldLabel, //1
49024                f.labelStyle||this.labelStyle||'', //2
49025                this.elementStyle||'', //3
49026                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49027                f.itemCls||this.itemCls||''  //5
49028        ], true).getPrevSibling());
49029     },
49030
49031     // private
49032     renderComponent : function(c){
49033         c.render(c.isLayout ? this.el : this.el.createChild());    
49034     },
49035     /**
49036      * Adds a object form elements (using the xtype property as the factory method.)
49037      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49038      * @param {Object} config 
49039      */
49040     addxtype : function(o)
49041     {
49042         // create the lement.
49043         o.form = this.form;
49044         var fe = Roo.factory(o, Roo.form);
49045         this.form.allItems.push(fe);
49046         this.stack.push(fe);
49047         
49048         if (fe.isFormField) {
49049             this.form.items.add(fe);
49050         }
49051          
49052         return fe;
49053     }
49054 });
49055
49056 /**
49057  * @class Roo.form.Column
49058  * @extends Roo.form.Layout
49059  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49060  * @constructor
49061  * @param {Object} config Configuration options
49062  */
49063 Roo.form.Column = function(config){
49064     Roo.form.Column.superclass.constructor.call(this, config);
49065 };
49066
49067 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49068     /**
49069      * @cfg {Number/String} width
49070      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49071      */
49072     /**
49073      * @cfg {String/Object} autoCreate
49074      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49075      */
49076
49077     // private
49078     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49079
49080     // private
49081     onRender : function(ct, position){
49082         Roo.form.Column.superclass.onRender.call(this, ct, position);
49083         if(this.width){
49084             this.el.setWidth(this.width);
49085         }
49086     }
49087 });
49088
49089
49090 /**
49091  * @class Roo.form.Row
49092  * @extends Roo.form.Layout
49093  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49094  * @constructor
49095  * @param {Object} config Configuration options
49096  */
49097
49098  
49099 Roo.form.Row = function(config){
49100     Roo.form.Row.superclass.constructor.call(this, config);
49101 };
49102  
49103 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49104       /**
49105      * @cfg {Number/String} width
49106      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49107      */
49108     /**
49109      * @cfg {Number/String} height
49110      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49111      */
49112     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49113     
49114     padWidth : 20,
49115     // private
49116     onRender : function(ct, position){
49117         //console.log('row render');
49118         if(!this.rowTpl){
49119             var t = new Roo.Template(
49120                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49121                     '<label for="{0}" style="{2}">{1}{4}</label>',
49122                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49123                     '</div>',
49124                 '</div>'
49125             );
49126             t.disableFormats = true;
49127             t.compile();
49128             Roo.form.Layout.prototype.rowTpl = t;
49129         }
49130         this.fieldTpl = this.rowTpl;
49131         
49132         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49133         var labelWidth = 100;
49134         
49135         if ((this.labelAlign != 'top')) {
49136             if (typeof this.labelWidth == 'number') {
49137                 labelWidth = this.labelWidth
49138             }
49139             this.padWidth =  20 + labelWidth;
49140             
49141         }
49142         
49143         Roo.form.Column.superclass.onRender.call(this, ct, position);
49144         if(this.width){
49145             this.el.setWidth(this.width);
49146         }
49147         if(this.height){
49148             this.el.setHeight(this.height);
49149         }
49150     },
49151     
49152     // private
49153     renderField : function(f){
49154         f.fieldEl = this.fieldTpl.append(this.el, [
49155                f.id, f.fieldLabel,
49156                f.labelStyle||this.labelStyle||'',
49157                this.elementStyle||'',
49158                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49159                f.itemCls||this.itemCls||'',
49160                f.width ? f.width + this.padWidth : 160 + this.padWidth
49161        ],true);
49162     }
49163 });
49164  
49165
49166 /**
49167  * @class Roo.form.FieldSet
49168  * @extends Roo.form.Layout
49169  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49170  * @constructor
49171  * @param {Object} config Configuration options
49172  */
49173 Roo.form.FieldSet = function(config){
49174     Roo.form.FieldSet.superclass.constructor.call(this, config);
49175 };
49176
49177 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49178     /**
49179      * @cfg {String} legend
49180      * The text to display as the legend for the FieldSet (defaults to '')
49181      */
49182     /**
49183      * @cfg {String/Object} autoCreate
49184      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49185      */
49186
49187     // private
49188     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49189
49190     // private
49191     onRender : function(ct, position){
49192         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49193         if(this.legend){
49194             this.setLegend(this.legend);
49195         }
49196     },
49197
49198     // private
49199     setLegend : function(text){
49200         if(this.rendered){
49201             this.el.child('legend').update(text);
49202         }
49203     }
49204 });/*
49205  * Based on:
49206  * Ext JS Library 1.1.1
49207  * Copyright(c) 2006-2007, Ext JS, LLC.
49208  *
49209  * Originally Released Under LGPL - original licence link has changed is not relivant.
49210  *
49211  * Fork - LGPL
49212  * <script type="text/javascript">
49213  */
49214 /**
49215  * @class Roo.form.VTypes
49216  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49217  * @singleton
49218  */
49219 Roo.form.VTypes = function(){
49220     // closure these in so they are only created once.
49221     var alpha = /^[a-zA-Z_]+$/;
49222     var alphanum = /^[a-zA-Z0-9_]+$/;
49223     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49224     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49225
49226     // All these messages and functions are configurable
49227     return {
49228         /**
49229          * The function used to validate email addresses
49230          * @param {String} value The email address
49231          */
49232         'email' : function(v){
49233             return email.test(v);
49234         },
49235         /**
49236          * The error text to display when the email validation function returns false
49237          * @type String
49238          */
49239         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49240         /**
49241          * The keystroke filter mask to be applied on email input
49242          * @type RegExp
49243          */
49244         'emailMask' : /[a-z0-9_\.\-@]/i,
49245
49246         /**
49247          * The function used to validate URLs
49248          * @param {String} value The URL
49249          */
49250         'url' : function(v){
49251             return url.test(v);
49252         },
49253         /**
49254          * The error text to display when the url validation function returns false
49255          * @type String
49256          */
49257         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49258         
49259         /**
49260          * The function used to validate alpha values
49261          * @param {String} value The value
49262          */
49263         'alpha' : function(v){
49264             return alpha.test(v);
49265         },
49266         /**
49267          * The error text to display when the alpha validation function returns false
49268          * @type String
49269          */
49270         'alphaText' : 'This field should only contain letters and _',
49271         /**
49272          * The keystroke filter mask to be applied on alpha input
49273          * @type RegExp
49274          */
49275         'alphaMask' : /[a-z_]/i,
49276
49277         /**
49278          * The function used to validate alphanumeric values
49279          * @param {String} value The value
49280          */
49281         'alphanum' : function(v){
49282             return alphanum.test(v);
49283         },
49284         /**
49285          * The error text to display when the alphanumeric validation function returns false
49286          * @type String
49287          */
49288         'alphanumText' : 'This field should only contain letters, numbers and _',
49289         /**
49290          * The keystroke filter mask to be applied on alphanumeric input
49291          * @type RegExp
49292          */
49293         'alphanumMask' : /[a-z0-9_]/i
49294     };
49295 }();//<script type="text/javascript">
49296
49297 /**
49298  * @class Roo.form.FCKeditor
49299  * @extends Roo.form.TextArea
49300  * Wrapper around the FCKEditor http://www.fckeditor.net
49301  * @constructor
49302  * Creates a new FCKeditor
49303  * @param {Object} config Configuration options
49304  */
49305 Roo.form.FCKeditor = function(config){
49306     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49307     this.addEvents({
49308          /**
49309          * @event editorinit
49310          * Fired when the editor is initialized - you can add extra handlers here..
49311          * @param {FCKeditor} this
49312          * @param {Object} the FCK object.
49313          */
49314         editorinit : true
49315     });
49316     
49317     
49318 };
49319 Roo.form.FCKeditor.editors = { };
49320 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49321 {
49322     //defaultAutoCreate : {
49323     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49324     //},
49325     // private
49326     /**
49327      * @cfg {Object} fck options - see fck manual for details.
49328      */
49329     fckconfig : false,
49330     
49331     /**
49332      * @cfg {Object} fck toolbar set (Basic or Default)
49333      */
49334     toolbarSet : 'Basic',
49335     /**
49336      * @cfg {Object} fck BasePath
49337      */ 
49338     basePath : '/fckeditor/',
49339     
49340     
49341     frame : false,
49342     
49343     value : '',
49344     
49345    
49346     onRender : function(ct, position)
49347     {
49348         if(!this.el){
49349             this.defaultAutoCreate = {
49350                 tag: "textarea",
49351                 style:"width:300px;height:60px;",
49352                 autocomplete: "new-password"
49353             };
49354         }
49355         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49356         /*
49357         if(this.grow){
49358             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49359             if(this.preventScrollbars){
49360                 this.el.setStyle("overflow", "hidden");
49361             }
49362             this.el.setHeight(this.growMin);
49363         }
49364         */
49365         //console.log('onrender' + this.getId() );
49366         Roo.form.FCKeditor.editors[this.getId()] = this;
49367          
49368
49369         this.replaceTextarea() ;
49370         
49371     },
49372     
49373     getEditor : function() {
49374         return this.fckEditor;
49375     },
49376     /**
49377      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49378      * @param {Mixed} value The value to set
49379      */
49380     
49381     
49382     setValue : function(value)
49383     {
49384         //console.log('setValue: ' + value);
49385         
49386         if(typeof(value) == 'undefined') { // not sure why this is happending...
49387             return;
49388         }
49389         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49390         
49391         //if(!this.el || !this.getEditor()) {
49392         //    this.value = value;
49393             //this.setValue.defer(100,this,[value]);    
49394         //    return;
49395         //} 
49396         
49397         if(!this.getEditor()) {
49398             return;
49399         }
49400         
49401         this.getEditor().SetData(value);
49402         
49403         //
49404
49405     },
49406
49407     /**
49408      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49409      * @return {Mixed} value The field value
49410      */
49411     getValue : function()
49412     {
49413         
49414         if (this.frame && this.frame.dom.style.display == 'none') {
49415             return Roo.form.FCKeditor.superclass.getValue.call(this);
49416         }
49417         
49418         if(!this.el || !this.getEditor()) {
49419            
49420            // this.getValue.defer(100,this); 
49421             return this.value;
49422         }
49423        
49424         
49425         var value=this.getEditor().GetData();
49426         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49427         return Roo.form.FCKeditor.superclass.getValue.call(this);
49428         
49429
49430     },
49431
49432     /**
49433      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49434      * @return {Mixed} value The field value
49435      */
49436     getRawValue : function()
49437     {
49438         if (this.frame && this.frame.dom.style.display == 'none') {
49439             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49440         }
49441         
49442         if(!this.el || !this.getEditor()) {
49443             //this.getRawValue.defer(100,this); 
49444             return this.value;
49445             return;
49446         }
49447         
49448         
49449         
49450         var value=this.getEditor().GetData();
49451         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49452         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49453          
49454     },
49455     
49456     setSize : function(w,h) {
49457         
49458         
49459         
49460         //if (this.frame && this.frame.dom.style.display == 'none') {
49461         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49462         //    return;
49463         //}
49464         //if(!this.el || !this.getEditor()) {
49465         //    this.setSize.defer(100,this, [w,h]); 
49466         //    return;
49467         //}
49468         
49469         
49470         
49471         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49472         
49473         this.frame.dom.setAttribute('width', w);
49474         this.frame.dom.setAttribute('height', h);
49475         this.frame.setSize(w,h);
49476         
49477     },
49478     
49479     toggleSourceEdit : function(value) {
49480         
49481       
49482          
49483         this.el.dom.style.display = value ? '' : 'none';
49484         this.frame.dom.style.display = value ?  'none' : '';
49485         
49486     },
49487     
49488     
49489     focus: function(tag)
49490     {
49491         if (this.frame.dom.style.display == 'none') {
49492             return Roo.form.FCKeditor.superclass.focus.call(this);
49493         }
49494         if(!this.el || !this.getEditor()) {
49495             this.focus.defer(100,this, [tag]); 
49496             return;
49497         }
49498         
49499         
49500         
49501         
49502         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49503         this.getEditor().Focus();
49504         if (tgs.length) {
49505             if (!this.getEditor().Selection.GetSelection()) {
49506                 this.focus.defer(100,this, [tag]); 
49507                 return;
49508             }
49509             
49510             
49511             var r = this.getEditor().EditorDocument.createRange();
49512             r.setStart(tgs[0],0);
49513             r.setEnd(tgs[0],0);
49514             this.getEditor().Selection.GetSelection().removeAllRanges();
49515             this.getEditor().Selection.GetSelection().addRange(r);
49516             this.getEditor().Focus();
49517         }
49518         
49519     },
49520     
49521     
49522     
49523     replaceTextarea : function()
49524     {
49525         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49526             return ;
49527         }
49528         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49529         //{
49530             // We must check the elements firstly using the Id and then the name.
49531         var oTextarea = document.getElementById( this.getId() );
49532         
49533         var colElementsByName = document.getElementsByName( this.getId() ) ;
49534          
49535         oTextarea.style.display = 'none' ;
49536
49537         if ( oTextarea.tabIndex ) {            
49538             this.TabIndex = oTextarea.tabIndex ;
49539         }
49540         
49541         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49542         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49543         this.frame = Roo.get(this.getId() + '___Frame')
49544     },
49545     
49546     _getConfigHtml : function()
49547     {
49548         var sConfig = '' ;
49549
49550         for ( var o in this.fckconfig ) {
49551             sConfig += sConfig.length > 0  ? '&amp;' : '';
49552             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49553         }
49554
49555         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49556     },
49557     
49558     
49559     _getIFrameHtml : function()
49560     {
49561         var sFile = 'fckeditor.html' ;
49562         /* no idea what this is about..
49563         try
49564         {
49565             if ( (/fcksource=true/i).test( window.top.location.search ) )
49566                 sFile = 'fckeditor.original.html' ;
49567         }
49568         catch (e) { 
49569         */
49570
49571         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49572         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49573         
49574         
49575         var html = '<iframe id="' + this.getId() +
49576             '___Frame" src="' + sLink +
49577             '" width="' + this.width +
49578             '" height="' + this.height + '"' +
49579             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49580             ' frameborder="0" scrolling="no"></iframe>' ;
49581
49582         return html ;
49583     },
49584     
49585     _insertHtmlBefore : function( html, element )
49586     {
49587         if ( element.insertAdjacentHTML )       {
49588             // IE
49589             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49590         } else { // Gecko
49591             var oRange = document.createRange() ;
49592             oRange.setStartBefore( element ) ;
49593             var oFragment = oRange.createContextualFragment( html );
49594             element.parentNode.insertBefore( oFragment, element ) ;
49595         }
49596     }
49597     
49598     
49599   
49600     
49601     
49602     
49603     
49604
49605 });
49606
49607 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49608
49609 function FCKeditor_OnComplete(editorInstance){
49610     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49611     f.fckEditor = editorInstance;
49612     //console.log("loaded");
49613     f.fireEvent('editorinit', f, editorInstance);
49614
49615   
49616
49617  
49618
49619
49620
49621
49622
49623
49624
49625
49626
49627
49628
49629
49630
49631
49632
49633 //<script type="text/javascript">
49634 /**
49635  * @class Roo.form.GridField
49636  * @extends Roo.form.Field
49637  * Embed a grid (or editable grid into a form)
49638  * STATUS ALPHA
49639  * 
49640  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49641  * it needs 
49642  * xgrid.store = Roo.data.Store
49643  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49644  * xgrid.store.reader = Roo.data.JsonReader 
49645  * 
49646  * 
49647  * @constructor
49648  * Creates a new GridField
49649  * @param {Object} config Configuration options
49650  */
49651 Roo.form.GridField = function(config){
49652     Roo.form.GridField.superclass.constructor.call(this, config);
49653      
49654 };
49655
49656 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49657     /**
49658      * @cfg {Number} width  - used to restrict width of grid..
49659      */
49660     width : 100,
49661     /**
49662      * @cfg {Number} height - used to restrict height of grid..
49663      */
49664     height : 50,
49665      /**
49666      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49667          * 
49668          *}
49669      */
49670     xgrid : false, 
49671     /**
49672      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49673      * {tag: "input", type: "checkbox", autocomplete: "off"})
49674      */
49675    // defaultAutoCreate : { tag: 'div' },
49676     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49677     /**
49678      * @cfg {String} addTitle Text to include for adding a title.
49679      */
49680     addTitle : false,
49681     //
49682     onResize : function(){
49683         Roo.form.Field.superclass.onResize.apply(this, arguments);
49684     },
49685
49686     initEvents : function(){
49687         // Roo.form.Checkbox.superclass.initEvents.call(this);
49688         // has no events...
49689        
49690     },
49691
49692
49693     getResizeEl : function(){
49694         return this.wrap;
49695     },
49696
49697     getPositionEl : function(){
49698         return this.wrap;
49699     },
49700
49701     // private
49702     onRender : function(ct, position){
49703         
49704         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49705         var style = this.style;
49706         delete this.style;
49707         
49708         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49709         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49710         this.viewEl = this.wrap.createChild({ tag: 'div' });
49711         if (style) {
49712             this.viewEl.applyStyles(style);
49713         }
49714         if (this.width) {
49715             this.viewEl.setWidth(this.width);
49716         }
49717         if (this.height) {
49718             this.viewEl.setHeight(this.height);
49719         }
49720         //if(this.inputValue !== undefined){
49721         //this.setValue(this.value);
49722         
49723         
49724         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49725         
49726         
49727         this.grid.render();
49728         this.grid.getDataSource().on('remove', this.refreshValue, this);
49729         this.grid.getDataSource().on('update', this.refreshValue, this);
49730         this.grid.on('afteredit', this.refreshValue, this);
49731  
49732     },
49733      
49734     
49735     /**
49736      * Sets the value of the item. 
49737      * @param {String} either an object  or a string..
49738      */
49739     setValue : function(v){
49740         //this.value = v;
49741         v = v || []; // empty set..
49742         // this does not seem smart - it really only affects memoryproxy grids..
49743         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49744             var ds = this.grid.getDataSource();
49745             // assumes a json reader..
49746             var data = {}
49747             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49748             ds.loadData( data);
49749         }
49750         // clear selection so it does not get stale.
49751         if (this.grid.sm) { 
49752             this.grid.sm.clearSelections();
49753         }
49754         
49755         Roo.form.GridField.superclass.setValue.call(this, v);
49756         this.refreshValue();
49757         // should load data in the grid really....
49758     },
49759     
49760     // private
49761     refreshValue: function() {
49762          var val = [];
49763         this.grid.getDataSource().each(function(r) {
49764             val.push(r.data);
49765         });
49766         this.el.dom.value = Roo.encode(val);
49767     }
49768     
49769      
49770     
49771     
49772 });/*
49773  * Based on:
49774  * Ext JS Library 1.1.1
49775  * Copyright(c) 2006-2007, Ext JS, LLC.
49776  *
49777  * Originally Released Under LGPL - original licence link has changed is not relivant.
49778  *
49779  * Fork - LGPL
49780  * <script type="text/javascript">
49781  */
49782 /**
49783  * @class Roo.form.DisplayField
49784  * @extends Roo.form.Field
49785  * A generic Field to display non-editable data.
49786  * @cfg {Boolean} closable (true|false) default false
49787  * @constructor
49788  * Creates a new Display Field item.
49789  * @param {Object} config Configuration options
49790  */
49791 Roo.form.DisplayField = function(config){
49792     Roo.form.DisplayField.superclass.constructor.call(this, config);
49793     
49794     this.addEvents({
49795         /**
49796          * @event close
49797          * Fires after the click the close btn
49798              * @param {Roo.form.DisplayField} this
49799              */
49800         close : true
49801     });
49802 };
49803
49804 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49805     inputType:      'hidden',
49806     allowBlank:     true,
49807     readOnly:         true,
49808     
49809  
49810     /**
49811      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49812      */
49813     focusClass : undefined,
49814     /**
49815      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49816      */
49817     fieldClass: 'x-form-field',
49818     
49819      /**
49820      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49821      */
49822     valueRenderer: undefined,
49823     
49824     width: 100,
49825     /**
49826      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49827      * {tag: "input", type: "checkbox", autocomplete: "off"})
49828      */
49829      
49830  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49831  
49832     closable : false,
49833     
49834     onResize : function(){
49835         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49836         
49837     },
49838
49839     initEvents : function(){
49840         // Roo.form.Checkbox.superclass.initEvents.call(this);
49841         // has no events...
49842         
49843         if(this.closable){
49844             this.closeEl.on('click', this.onClose, this);
49845         }
49846        
49847     },
49848
49849
49850     getResizeEl : function(){
49851         return this.wrap;
49852     },
49853
49854     getPositionEl : function(){
49855         return this.wrap;
49856     },
49857
49858     // private
49859     onRender : function(ct, position){
49860         
49861         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49862         //if(this.inputValue !== undefined){
49863         this.wrap = this.el.wrap();
49864         
49865         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49866         
49867         if(this.closable){
49868             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49869         }
49870         
49871         if (this.bodyStyle) {
49872             this.viewEl.applyStyles(this.bodyStyle);
49873         }
49874         //this.viewEl.setStyle('padding', '2px');
49875         
49876         this.setValue(this.value);
49877         
49878     },
49879 /*
49880     // private
49881     initValue : Roo.emptyFn,
49882
49883   */
49884
49885         // private
49886     onClick : function(){
49887         
49888     },
49889
49890     /**
49891      * Sets the checked state of the checkbox.
49892      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49893      */
49894     setValue : function(v){
49895         this.value = v;
49896         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49897         // this might be called before we have a dom element..
49898         if (!this.viewEl) {
49899             return;
49900         }
49901         this.viewEl.dom.innerHTML = html;
49902         Roo.form.DisplayField.superclass.setValue.call(this, v);
49903
49904     },
49905     
49906     onClose : function(e)
49907     {
49908         e.preventDefault();
49909         
49910         this.fireEvent('close', this);
49911     }
49912 });/*
49913  * 
49914  * Licence- LGPL
49915  * 
49916  */
49917
49918 /**
49919  * @class Roo.form.DayPicker
49920  * @extends Roo.form.Field
49921  * A Day picker show [M] [T] [W] ....
49922  * @constructor
49923  * Creates a new Day Picker
49924  * @param {Object} config Configuration options
49925  */
49926 Roo.form.DayPicker= function(config){
49927     Roo.form.DayPicker.superclass.constructor.call(this, config);
49928      
49929 };
49930
49931 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49932     /**
49933      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49934      */
49935     focusClass : undefined,
49936     /**
49937      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49938      */
49939     fieldClass: "x-form-field",
49940    
49941     /**
49942      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49943      * {tag: "input", type: "checkbox", autocomplete: "off"})
49944      */
49945     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49946     
49947    
49948     actionMode : 'viewEl', 
49949     //
49950     // private
49951  
49952     inputType : 'hidden',
49953     
49954      
49955     inputElement: false, // real input element?
49956     basedOn: false, // ????
49957     
49958     isFormField: true, // not sure where this is needed!!!!
49959
49960     onResize : function(){
49961         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49962         if(!this.boxLabel){
49963             this.el.alignTo(this.wrap, 'c-c');
49964         }
49965     },
49966
49967     initEvents : function(){
49968         Roo.form.Checkbox.superclass.initEvents.call(this);
49969         this.el.on("click", this.onClick,  this);
49970         this.el.on("change", this.onClick,  this);
49971     },
49972
49973
49974     getResizeEl : function(){
49975         return this.wrap;
49976     },
49977
49978     getPositionEl : function(){
49979         return this.wrap;
49980     },
49981
49982     
49983     // private
49984     onRender : function(ct, position){
49985         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49986        
49987         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49988         
49989         var r1 = '<table><tr>';
49990         var r2 = '<tr class="x-form-daypick-icons">';
49991         for (var i=0; i < 7; i++) {
49992             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49993             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49994         }
49995         
49996         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49997         viewEl.select('img').on('click', this.onClick, this);
49998         this.viewEl = viewEl;   
49999         
50000         
50001         // this will not work on Chrome!!!
50002         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
50003         this.el.on('propertychange', this.setFromHidden,  this);  //ie
50004         
50005         
50006           
50007
50008     },
50009
50010     // private
50011     initValue : Roo.emptyFn,
50012
50013     /**
50014      * Returns the checked state of the checkbox.
50015      * @return {Boolean} True if checked, else false
50016      */
50017     getValue : function(){
50018         return this.el.dom.value;
50019         
50020     },
50021
50022         // private
50023     onClick : function(e){ 
50024         //this.setChecked(!this.checked);
50025         Roo.get(e.target).toggleClass('x-menu-item-checked');
50026         this.refreshValue();
50027         //if(this.el.dom.checked != this.checked){
50028         //    this.setValue(this.el.dom.checked);
50029        // }
50030     },
50031     
50032     // private
50033     refreshValue : function()
50034     {
50035         var val = '';
50036         this.viewEl.select('img',true).each(function(e,i,n)  {
50037             val += e.is(".x-menu-item-checked") ? String(n) : '';
50038         });
50039         this.setValue(val, true);
50040     },
50041
50042     /**
50043      * Sets the checked state of the checkbox.
50044      * On is always based on a string comparison between inputValue and the param.
50045      * @param {Boolean/String} value - the value to set 
50046      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50047      */
50048     setValue : function(v,suppressEvent){
50049         if (!this.el.dom) {
50050             return;
50051         }
50052         var old = this.el.dom.value ;
50053         this.el.dom.value = v;
50054         if (suppressEvent) {
50055             return ;
50056         }
50057          
50058         // update display..
50059         this.viewEl.select('img',true).each(function(e,i,n)  {
50060             
50061             var on = e.is(".x-menu-item-checked");
50062             var newv = v.indexOf(String(n)) > -1;
50063             if (on != newv) {
50064                 e.toggleClass('x-menu-item-checked');
50065             }
50066             
50067         });
50068         
50069         
50070         this.fireEvent('change', this, v, old);
50071         
50072         
50073     },
50074    
50075     // handle setting of hidden value by some other method!!?!?
50076     setFromHidden: function()
50077     {
50078         if(!this.el){
50079             return;
50080         }
50081         //console.log("SET FROM HIDDEN");
50082         //alert('setFrom hidden');
50083         this.setValue(this.el.dom.value);
50084     },
50085     
50086     onDestroy : function()
50087     {
50088         if(this.viewEl){
50089             Roo.get(this.viewEl).remove();
50090         }
50091          
50092         Roo.form.DayPicker.superclass.onDestroy.call(this);
50093     }
50094
50095 });/*
50096  * RooJS Library 1.1.1
50097  * Copyright(c) 2008-2011  Alan Knowles
50098  *
50099  * License - LGPL
50100  */
50101  
50102
50103 /**
50104  * @class Roo.form.ComboCheck
50105  * @extends Roo.form.ComboBox
50106  * A combobox for multiple select items.
50107  *
50108  * FIXME - could do with a reset button..
50109  * 
50110  * @constructor
50111  * Create a new ComboCheck
50112  * @param {Object} config Configuration options
50113  */
50114 Roo.form.ComboCheck = function(config){
50115     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50116     // should verify some data...
50117     // like
50118     // hiddenName = required..
50119     // displayField = required
50120     // valudField == required
50121     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50122     var _t = this;
50123     Roo.each(req, function(e) {
50124         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50125             throw "Roo.form.ComboCheck : missing value for: " + e;
50126         }
50127     });
50128     
50129     
50130 };
50131
50132 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50133      
50134      
50135     editable : false,
50136      
50137     selectedClass: 'x-menu-item-checked', 
50138     
50139     // private
50140     onRender : function(ct, position){
50141         var _t = this;
50142         
50143         
50144         
50145         if(!this.tpl){
50146             var cls = 'x-combo-list';
50147
50148             
50149             this.tpl =  new Roo.Template({
50150                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50151                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50152                    '<span>{' + this.displayField + '}</span>' +
50153                     '</div>' 
50154                 
50155             });
50156         }
50157  
50158         
50159         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50160         this.view.singleSelect = false;
50161         this.view.multiSelect = true;
50162         this.view.toggleSelect = true;
50163         this.pageTb.add(new Roo.Toolbar.Fill(), {
50164             
50165             text: 'Done',
50166             handler: function()
50167             {
50168                 _t.collapse();
50169             }
50170         });
50171     },
50172     
50173     onViewOver : function(e, t){
50174         // do nothing...
50175         return;
50176         
50177     },
50178     
50179     onViewClick : function(doFocus,index){
50180         return;
50181         
50182     },
50183     select: function () {
50184         //Roo.log("SELECT CALLED");
50185     },
50186      
50187     selectByValue : function(xv, scrollIntoView){
50188         var ar = this.getValueArray();
50189         var sels = [];
50190         
50191         Roo.each(ar, function(v) {
50192             if(v === undefined || v === null){
50193                 return;
50194             }
50195             var r = this.findRecord(this.valueField, v);
50196             if(r){
50197                 sels.push(this.store.indexOf(r))
50198                 
50199             }
50200         },this);
50201         this.view.select(sels);
50202         return false;
50203     },
50204     
50205     
50206     
50207     onSelect : function(record, index){
50208        // Roo.log("onselect Called");
50209        // this is only called by the clear button now..
50210         this.view.clearSelections();
50211         this.setValue('[]');
50212         if (this.value != this.valueBefore) {
50213             this.fireEvent('change', this, this.value, this.valueBefore);
50214             this.valueBefore = this.value;
50215         }
50216     },
50217     getValueArray : function()
50218     {
50219         var ar = [] ;
50220         
50221         try {
50222             //Roo.log(this.value);
50223             if (typeof(this.value) == 'undefined') {
50224                 return [];
50225             }
50226             var ar = Roo.decode(this.value);
50227             return  ar instanceof Array ? ar : []; //?? valid?
50228             
50229         } catch(e) {
50230             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50231             return [];
50232         }
50233          
50234     },
50235     expand : function ()
50236     {
50237         
50238         Roo.form.ComboCheck.superclass.expand.call(this);
50239         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50240         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50241         
50242
50243     },
50244     
50245     collapse : function(){
50246         Roo.form.ComboCheck.superclass.collapse.call(this);
50247         var sl = this.view.getSelectedIndexes();
50248         var st = this.store;
50249         var nv = [];
50250         var tv = [];
50251         var r;
50252         Roo.each(sl, function(i) {
50253             r = st.getAt(i);
50254             nv.push(r.get(this.valueField));
50255         },this);
50256         this.setValue(Roo.encode(nv));
50257         if (this.value != this.valueBefore) {
50258
50259             this.fireEvent('change', this, this.value, this.valueBefore);
50260             this.valueBefore = this.value;
50261         }
50262         
50263     },
50264     
50265     setValue : function(v){
50266         // Roo.log(v);
50267         this.value = v;
50268         
50269         var vals = this.getValueArray();
50270         var tv = [];
50271         Roo.each(vals, function(k) {
50272             var r = this.findRecord(this.valueField, k);
50273             if(r){
50274                 tv.push(r.data[this.displayField]);
50275             }else if(this.valueNotFoundText !== undefined){
50276                 tv.push( this.valueNotFoundText );
50277             }
50278         },this);
50279        // Roo.log(tv);
50280         
50281         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50282         this.hiddenField.value = v;
50283         this.value = v;
50284     }
50285     
50286 });/*
50287  * Based on:
50288  * Ext JS Library 1.1.1
50289  * Copyright(c) 2006-2007, Ext JS, LLC.
50290  *
50291  * Originally Released Under LGPL - original licence link has changed is not relivant.
50292  *
50293  * Fork - LGPL
50294  * <script type="text/javascript">
50295  */
50296  
50297 /**
50298  * @class Roo.form.Signature
50299  * @extends Roo.form.Field
50300  * Signature field.  
50301  * @constructor
50302  * 
50303  * @param {Object} config Configuration options
50304  */
50305
50306 Roo.form.Signature = function(config){
50307     Roo.form.Signature.superclass.constructor.call(this, config);
50308     
50309     this.addEvents({// not in used??
50310          /**
50311          * @event confirm
50312          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50313              * @param {Roo.form.Signature} combo This combo box
50314              */
50315         'confirm' : true,
50316         /**
50317          * @event reset
50318          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50319              * @param {Roo.form.ComboBox} combo This combo box
50320              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50321              */
50322         'reset' : true
50323     });
50324 };
50325
50326 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50327     /**
50328      * @cfg {Object} labels Label to use when rendering a form.
50329      * defaults to 
50330      * labels : { 
50331      *      clear : "Clear",
50332      *      confirm : "Confirm"
50333      *  }
50334      */
50335     labels : { 
50336         clear : "Clear",
50337         confirm : "Confirm"
50338     },
50339     /**
50340      * @cfg {Number} width The signature panel width (defaults to 300)
50341      */
50342     width: 300,
50343     /**
50344      * @cfg {Number} height The signature panel height (defaults to 100)
50345      */
50346     height : 100,
50347     /**
50348      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50349      */
50350     allowBlank : false,
50351     
50352     //private
50353     // {Object} signPanel The signature SVG panel element (defaults to {})
50354     signPanel : {},
50355     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50356     isMouseDown : false,
50357     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50358     isConfirmed : false,
50359     // {String} signatureTmp SVG mapping string (defaults to empty string)
50360     signatureTmp : '',
50361     
50362     
50363     defaultAutoCreate : { // modified by initCompnoent..
50364         tag: "input",
50365         type:"hidden"
50366     },
50367
50368     // private
50369     onRender : function(ct, position){
50370         
50371         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50372         
50373         this.wrap = this.el.wrap({
50374             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50375         });
50376         
50377         this.createToolbar(this);
50378         this.signPanel = this.wrap.createChild({
50379                 tag: 'div',
50380                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50381             }, this.el
50382         );
50383             
50384         this.svgID = Roo.id();
50385         this.svgEl = this.signPanel.createChild({
50386               xmlns : 'http://www.w3.org/2000/svg',
50387               tag : 'svg',
50388               id : this.svgID + "-svg",
50389               width: this.width,
50390               height: this.height,
50391               viewBox: '0 0 '+this.width+' '+this.height,
50392               cn : [
50393                 {
50394                     tag: "rect",
50395                     id: this.svgID + "-svg-r",
50396                     width: this.width,
50397                     height: this.height,
50398                     fill: "#ffa"
50399                 },
50400                 {
50401                     tag: "line",
50402                     id: this.svgID + "-svg-l",
50403                     x1: "0", // start
50404                     y1: (this.height*0.8), // start set the line in 80% of height
50405                     x2: this.width, // end
50406                     y2: (this.height*0.8), // end set the line in 80% of height
50407                     'stroke': "#666",
50408                     'stroke-width': "1",
50409                     'stroke-dasharray': "3",
50410                     'shape-rendering': "crispEdges",
50411                     'pointer-events': "none"
50412                 },
50413                 {
50414                     tag: "path",
50415                     id: this.svgID + "-svg-p",
50416                     'stroke': "navy",
50417                     'stroke-width': "3",
50418                     'fill': "none",
50419                     'pointer-events': 'none'
50420                 }
50421               ]
50422         });
50423         this.createSVG();
50424         this.svgBox = this.svgEl.dom.getScreenCTM();
50425     },
50426     createSVG : function(){ 
50427         var svg = this.signPanel;
50428         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50429         var t = this;
50430
50431         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50432         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50433         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50434         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50435         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50436         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50437         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50438         
50439     },
50440     isTouchEvent : function(e){
50441         return e.type.match(/^touch/);
50442     },
50443     getCoords : function (e) {
50444         var pt    = this.svgEl.dom.createSVGPoint();
50445         pt.x = e.clientX; 
50446         pt.y = e.clientY;
50447         if (this.isTouchEvent(e)) {
50448             pt.x =  e.targetTouches[0].clientX;
50449             pt.y = e.targetTouches[0].clientY;
50450         }
50451         var a = this.svgEl.dom.getScreenCTM();
50452         var b = a.inverse();
50453         var mx = pt.matrixTransform(b);
50454         return mx.x + ',' + mx.y;
50455     },
50456     //mouse event headler 
50457     down : function (e) {
50458         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50459         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50460         
50461         this.isMouseDown = true;
50462         
50463         e.preventDefault();
50464     },
50465     move : function (e) {
50466         if (this.isMouseDown) {
50467             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50468             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50469         }
50470         
50471         e.preventDefault();
50472     },
50473     up : function (e) {
50474         this.isMouseDown = false;
50475         var sp = this.signatureTmp.split(' ');
50476         
50477         if(sp.length > 1){
50478             if(!sp[sp.length-2].match(/^L/)){
50479                 sp.pop();
50480                 sp.pop();
50481                 sp.push("");
50482                 this.signatureTmp = sp.join(" ");
50483             }
50484         }
50485         if(this.getValue() != this.signatureTmp){
50486             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50487             this.isConfirmed = false;
50488         }
50489         e.preventDefault();
50490     },
50491     
50492     /**
50493      * Protected method that will not generally be called directly. It
50494      * is called when the editor creates its toolbar. Override this method if you need to
50495      * add custom toolbar buttons.
50496      * @param {HtmlEditor} editor
50497      */
50498     createToolbar : function(editor){
50499          function btn(id, toggle, handler){
50500             var xid = fid + '-'+ id ;
50501             return {
50502                 id : xid,
50503                 cmd : id,
50504                 cls : 'x-btn-icon x-edit-'+id,
50505                 enableToggle:toggle !== false,
50506                 scope: editor, // was editor...
50507                 handler:handler||editor.relayBtnCmd,
50508                 clickEvent:'mousedown',
50509                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50510                 tabIndex:-1
50511             };
50512         }
50513         
50514         
50515         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50516         this.tb = tb;
50517         this.tb.add(
50518            {
50519                 cls : ' x-signature-btn x-signature-'+id,
50520                 scope: editor, // was editor...
50521                 handler: this.reset,
50522                 clickEvent:'mousedown',
50523                 text: this.labels.clear
50524             },
50525             {
50526                  xtype : 'Fill',
50527                  xns: Roo.Toolbar
50528             }, 
50529             {
50530                 cls : '  x-signature-btn x-signature-'+id,
50531                 scope: editor, // was editor...
50532                 handler: this.confirmHandler,
50533                 clickEvent:'mousedown',
50534                 text: this.labels.confirm
50535             }
50536         );
50537     
50538     },
50539     //public
50540     /**
50541      * when user is clicked confirm then show this image.....
50542      * 
50543      * @return {String} Image Data URI
50544      */
50545     getImageDataURI : function(){
50546         var svg = this.svgEl.dom.parentNode.innerHTML;
50547         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50548         return src; 
50549     },
50550     /**
50551      * 
50552      * @return {Boolean} this.isConfirmed
50553      */
50554     getConfirmed : function(){
50555         return this.isConfirmed;
50556     },
50557     /**
50558      * 
50559      * @return {Number} this.width
50560      */
50561     getWidth : function(){
50562         return this.width;
50563     },
50564     /**
50565      * 
50566      * @return {Number} this.height
50567      */
50568     getHeight : function(){
50569         return this.height;
50570     },
50571     // private
50572     getSignature : function(){
50573         return this.signatureTmp;
50574     },
50575     // private
50576     reset : function(){
50577         this.signatureTmp = '';
50578         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50579         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50580         this.isConfirmed = false;
50581         Roo.form.Signature.superclass.reset.call(this);
50582     },
50583     setSignature : function(s){
50584         this.signatureTmp = s;
50585         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50586         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50587         this.setValue(s);
50588         this.isConfirmed = false;
50589         Roo.form.Signature.superclass.reset.call(this);
50590     }, 
50591     test : function(){
50592 //        Roo.log(this.signPanel.dom.contentWindow.up())
50593     },
50594     //private
50595     setConfirmed : function(){
50596         
50597         
50598         
50599 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50600     },
50601     // private
50602     confirmHandler : function(){
50603         if(!this.getSignature()){
50604             return;
50605         }
50606         
50607         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50608         this.setValue(this.getSignature());
50609         this.isConfirmed = true;
50610         
50611         this.fireEvent('confirm', this);
50612     },
50613     // private
50614     // Subclasses should provide the validation implementation by overriding this
50615     validateValue : function(value){
50616         if(this.allowBlank){
50617             return true;
50618         }
50619         
50620         if(this.isConfirmed){
50621             return true;
50622         }
50623         return false;
50624     }
50625 });/*
50626  * Based on:
50627  * Ext JS Library 1.1.1
50628  * Copyright(c) 2006-2007, Ext JS, LLC.
50629  *
50630  * Originally Released Under LGPL - original licence link has changed is not relivant.
50631  *
50632  * Fork - LGPL
50633  * <script type="text/javascript">
50634  */
50635  
50636
50637 /**
50638  * @class Roo.form.ComboBox
50639  * @extends Roo.form.TriggerField
50640  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50641  * @constructor
50642  * Create a new ComboBox.
50643  * @param {Object} config Configuration options
50644  */
50645 Roo.form.Select = function(config){
50646     Roo.form.Select.superclass.constructor.call(this, config);
50647      
50648 };
50649
50650 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50651     /**
50652      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50653      */
50654     /**
50655      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50656      * rendering into an Roo.Editor, defaults to false)
50657      */
50658     /**
50659      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50660      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50661      */
50662     /**
50663      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50664      */
50665     /**
50666      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50667      * the dropdown list (defaults to undefined, with no header element)
50668      */
50669
50670      /**
50671      * @cfg {String/Roo.Template} tpl The template to use to render the output
50672      */
50673      
50674     // private
50675     defaultAutoCreate : {tag: "select"  },
50676     /**
50677      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50678      */
50679     listWidth: undefined,
50680     /**
50681      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50682      * mode = 'remote' or 'text' if mode = 'local')
50683      */
50684     displayField: undefined,
50685     /**
50686      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50687      * mode = 'remote' or 'value' if mode = 'local'). 
50688      * Note: use of a valueField requires the user make a selection
50689      * in order for a value to be mapped.
50690      */
50691     valueField: undefined,
50692     
50693     
50694     /**
50695      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50696      * field's data value (defaults to the underlying DOM element's name)
50697      */
50698     hiddenName: undefined,
50699     /**
50700      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50701      */
50702     listClass: '',
50703     /**
50704      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50705      */
50706     selectedClass: 'x-combo-selected',
50707     /**
50708      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50709      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50710      * which displays a downward arrow icon).
50711      */
50712     triggerClass : 'x-form-arrow-trigger',
50713     /**
50714      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50715      */
50716     shadow:'sides',
50717     /**
50718      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50719      * anchor positions (defaults to 'tl-bl')
50720      */
50721     listAlign: 'tl-bl?',
50722     /**
50723      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50724      */
50725     maxHeight: 300,
50726     /**
50727      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50728      * query specified by the allQuery config option (defaults to 'query')
50729      */
50730     triggerAction: 'query',
50731     /**
50732      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50733      * (defaults to 4, does not apply if editable = false)
50734      */
50735     minChars : 4,
50736     /**
50737      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50738      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50739      */
50740     typeAhead: false,
50741     /**
50742      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50743      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50744      */
50745     queryDelay: 500,
50746     /**
50747      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50748      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50749      */
50750     pageSize: 0,
50751     /**
50752      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50753      * when editable = true (defaults to false)
50754      */
50755     selectOnFocus:false,
50756     /**
50757      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50758      */
50759     queryParam: 'query',
50760     /**
50761      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50762      * when mode = 'remote' (defaults to 'Loading...')
50763      */
50764     loadingText: 'Loading...',
50765     /**
50766      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50767      */
50768     resizable: false,
50769     /**
50770      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50771      */
50772     handleHeight : 8,
50773     /**
50774      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50775      * traditional select (defaults to true)
50776      */
50777     editable: true,
50778     /**
50779      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50780      */
50781     allQuery: '',
50782     /**
50783      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50784      */
50785     mode: 'remote',
50786     /**
50787      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50788      * listWidth has a higher value)
50789      */
50790     minListWidth : 70,
50791     /**
50792      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50793      * allow the user to set arbitrary text into the field (defaults to false)
50794      */
50795     forceSelection:false,
50796     /**
50797      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50798      * if typeAhead = true (defaults to 250)
50799      */
50800     typeAheadDelay : 250,
50801     /**
50802      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50803      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50804      */
50805     valueNotFoundText : undefined,
50806     
50807     /**
50808      * @cfg {String} defaultValue The value displayed after loading the store.
50809      */
50810     defaultValue: '',
50811     
50812     /**
50813      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50814      */
50815     blockFocus : false,
50816     
50817     /**
50818      * @cfg {Boolean} disableClear Disable showing of clear button.
50819      */
50820     disableClear : false,
50821     /**
50822      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50823      */
50824     alwaysQuery : false,
50825     
50826     //private
50827     addicon : false,
50828     editicon: false,
50829     
50830     // element that contains real text value.. (when hidden is used..)
50831      
50832     // private
50833     onRender : function(ct, position){
50834         Roo.form.Field.prototype.onRender.call(this, ct, position);
50835         
50836         if(this.store){
50837             this.store.on('beforeload', this.onBeforeLoad, this);
50838             this.store.on('load', this.onLoad, this);
50839             this.store.on('loadexception', this.onLoadException, this);
50840             this.store.load({});
50841         }
50842         
50843         
50844         
50845     },
50846
50847     // private
50848     initEvents : function(){
50849         //Roo.form.ComboBox.superclass.initEvents.call(this);
50850  
50851     },
50852
50853     onDestroy : function(){
50854        
50855         if(this.store){
50856             this.store.un('beforeload', this.onBeforeLoad, this);
50857             this.store.un('load', this.onLoad, this);
50858             this.store.un('loadexception', this.onLoadException, this);
50859         }
50860         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50861     },
50862
50863     // private
50864     fireKey : function(e){
50865         if(e.isNavKeyPress() && !this.list.isVisible()){
50866             this.fireEvent("specialkey", this, e);
50867         }
50868     },
50869
50870     // private
50871     onResize: function(w, h){
50872         
50873         return; 
50874     
50875         
50876     },
50877
50878     /**
50879      * Allow or prevent the user from directly editing the field text.  If false is passed,
50880      * the user will only be able to select from the items defined in the dropdown list.  This method
50881      * is the runtime equivalent of setting the 'editable' config option at config time.
50882      * @param {Boolean} value True to allow the user to directly edit the field text
50883      */
50884     setEditable : function(value){
50885          
50886     },
50887
50888     // private
50889     onBeforeLoad : function(){
50890         
50891         Roo.log("Select before load");
50892         return;
50893     
50894         this.innerList.update(this.loadingText ?
50895                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50896         //this.restrictHeight();
50897         this.selectedIndex = -1;
50898     },
50899
50900     // private
50901     onLoad : function(){
50902
50903     
50904         var dom = this.el.dom;
50905         dom.innerHTML = '';
50906          var od = dom.ownerDocument;
50907          
50908         if (this.emptyText) {
50909             var op = od.createElement('option');
50910             op.setAttribute('value', '');
50911             op.innerHTML = String.format('{0}', this.emptyText);
50912             dom.appendChild(op);
50913         }
50914         if(this.store.getCount() > 0){
50915            
50916             var vf = this.valueField;
50917             var df = this.displayField;
50918             this.store.data.each(function(r) {
50919                 // which colmsn to use... testing - cdoe / title..
50920                 var op = od.createElement('option');
50921                 op.setAttribute('value', r.data[vf]);
50922                 op.innerHTML = String.format('{0}', r.data[df]);
50923                 dom.appendChild(op);
50924             });
50925             if (typeof(this.defaultValue != 'undefined')) {
50926                 this.setValue(this.defaultValue);
50927             }
50928             
50929              
50930         }else{
50931             //this.onEmptyResults();
50932         }
50933         //this.el.focus();
50934     },
50935     // private
50936     onLoadException : function()
50937     {
50938         dom.innerHTML = '';
50939             
50940         Roo.log("Select on load exception");
50941         return;
50942     
50943         this.collapse();
50944         Roo.log(this.store.reader.jsonData);
50945         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50946             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50947         }
50948         
50949         
50950     },
50951     // private
50952     onTypeAhead : function(){
50953          
50954     },
50955
50956     // private
50957     onSelect : function(record, index){
50958         Roo.log('on select?');
50959         return;
50960         if(this.fireEvent('beforeselect', this, record, index) !== false){
50961             this.setFromData(index > -1 ? record.data : false);
50962             this.collapse();
50963             this.fireEvent('select', this, record, index);
50964         }
50965     },
50966
50967     /**
50968      * Returns the currently selected field value or empty string if no value is set.
50969      * @return {String} value The selected value
50970      */
50971     getValue : function(){
50972         var dom = this.el.dom;
50973         this.value = dom.options[dom.selectedIndex].value;
50974         return this.value;
50975         
50976     },
50977
50978     /**
50979      * Clears any text/value currently set in the field
50980      */
50981     clearValue : function(){
50982         this.value = '';
50983         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50984         
50985     },
50986
50987     /**
50988      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50989      * will be displayed in the field.  If the value does not match the data value of an existing item,
50990      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50991      * Otherwise the field will be blank (although the value will still be set).
50992      * @param {String} value The value to match
50993      */
50994     setValue : function(v){
50995         var d = this.el.dom;
50996         for (var i =0; i < d.options.length;i++) {
50997             if (v == d.options[i].value) {
50998                 d.selectedIndex = i;
50999                 this.value = v;
51000                 return;
51001             }
51002         }
51003         this.clearValue();
51004     },
51005     /**
51006      * @property {Object} the last set data for the element
51007      */
51008     
51009     lastData : false,
51010     /**
51011      * Sets the value of the field based on a object which is related to the record format for the store.
51012      * @param {Object} value the value to set as. or false on reset?
51013      */
51014     setFromData : function(o){
51015         Roo.log('setfrom data?');
51016          
51017         
51018         
51019     },
51020     // private
51021     reset : function(){
51022         this.clearValue();
51023     },
51024     // private
51025     findRecord : function(prop, value){
51026         
51027         return false;
51028     
51029         var record;
51030         if(this.store.getCount() > 0){
51031             this.store.each(function(r){
51032                 if(r.data[prop] == value){
51033                     record = r;
51034                     return false;
51035                 }
51036                 return true;
51037             });
51038         }
51039         return record;
51040     },
51041     
51042     getName: function()
51043     {
51044         // returns hidden if it's set..
51045         if (!this.rendered) {return ''};
51046         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51047         
51048     },
51049      
51050
51051     
51052
51053     // private
51054     onEmptyResults : function(){
51055         Roo.log('empty results');
51056         //this.collapse();
51057     },
51058
51059     /**
51060      * Returns true if the dropdown list is expanded, else false.
51061      */
51062     isExpanded : function(){
51063         return false;
51064     },
51065
51066     /**
51067      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51068      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51069      * @param {String} value The data value of the item to select
51070      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51071      * selected item if it is not currently in view (defaults to true)
51072      * @return {Boolean} True if the value matched an item in the list, else false
51073      */
51074     selectByValue : function(v, scrollIntoView){
51075         Roo.log('select By Value');
51076         return false;
51077     
51078         if(v !== undefined && v !== null){
51079             var r = this.findRecord(this.valueField || this.displayField, v);
51080             if(r){
51081                 this.select(this.store.indexOf(r), scrollIntoView);
51082                 return true;
51083             }
51084         }
51085         return false;
51086     },
51087
51088     /**
51089      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51090      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51091      * @param {Number} index The zero-based index of the list item to select
51092      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51093      * selected item if it is not currently in view (defaults to true)
51094      */
51095     select : function(index, scrollIntoView){
51096         Roo.log('select ');
51097         return  ;
51098         
51099         this.selectedIndex = index;
51100         this.view.select(index);
51101         if(scrollIntoView !== false){
51102             var el = this.view.getNode(index);
51103             if(el){
51104                 this.innerList.scrollChildIntoView(el, false);
51105             }
51106         }
51107     },
51108
51109       
51110
51111     // private
51112     validateBlur : function(){
51113         
51114         return;
51115         
51116     },
51117
51118     // private
51119     initQuery : function(){
51120         this.doQuery(this.getRawValue());
51121     },
51122
51123     // private
51124     doForce : function(){
51125         if(this.el.dom.value.length > 0){
51126             this.el.dom.value =
51127                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51128              
51129         }
51130     },
51131
51132     /**
51133      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51134      * query allowing the query action to be canceled if needed.
51135      * @param {String} query The SQL query to execute
51136      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51137      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51138      * saved in the current store (defaults to false)
51139      */
51140     doQuery : function(q, forceAll){
51141         
51142         Roo.log('doQuery?');
51143         if(q === undefined || q === null){
51144             q = '';
51145         }
51146         var qe = {
51147             query: q,
51148             forceAll: forceAll,
51149             combo: this,
51150             cancel:false
51151         };
51152         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51153             return false;
51154         }
51155         q = qe.query;
51156         forceAll = qe.forceAll;
51157         if(forceAll === true || (q.length >= this.minChars)){
51158             if(this.lastQuery != q || this.alwaysQuery){
51159                 this.lastQuery = q;
51160                 if(this.mode == 'local'){
51161                     this.selectedIndex = -1;
51162                     if(forceAll){
51163                         this.store.clearFilter();
51164                     }else{
51165                         this.store.filter(this.displayField, q);
51166                     }
51167                     this.onLoad();
51168                 }else{
51169                     this.store.baseParams[this.queryParam] = q;
51170                     this.store.load({
51171                         params: this.getParams(q)
51172                     });
51173                     this.expand();
51174                 }
51175             }else{
51176                 this.selectedIndex = -1;
51177                 this.onLoad();   
51178             }
51179         }
51180     },
51181
51182     // private
51183     getParams : function(q){
51184         var p = {};
51185         //p[this.queryParam] = q;
51186         if(this.pageSize){
51187             p.start = 0;
51188             p.limit = this.pageSize;
51189         }
51190         return p;
51191     },
51192
51193     /**
51194      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51195      */
51196     collapse : function(){
51197         
51198     },
51199
51200     // private
51201     collapseIf : function(e){
51202         
51203     },
51204
51205     /**
51206      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51207      */
51208     expand : function(){
51209         
51210     } ,
51211
51212     // private
51213      
51214
51215     /** 
51216     * @cfg {Boolean} grow 
51217     * @hide 
51218     */
51219     /** 
51220     * @cfg {Number} growMin 
51221     * @hide 
51222     */
51223     /** 
51224     * @cfg {Number} growMax 
51225     * @hide 
51226     */
51227     /**
51228      * @hide
51229      * @method autoSize
51230      */
51231     
51232     setWidth : function()
51233     {
51234         
51235     },
51236     getResizeEl : function(){
51237         return this.el;
51238     }
51239 });//<script type="text/javasscript">
51240  
51241
51242 /**
51243  * @class Roo.DDView
51244  * A DnD enabled version of Roo.View.
51245  * @param {Element/String} container The Element in which to create the View.
51246  * @param {String} tpl The template string used to create the markup for each element of the View
51247  * @param {Object} config The configuration properties. These include all the config options of
51248  * {@link Roo.View} plus some specific to this class.<br>
51249  * <p>
51250  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51251  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51252  * <p>
51253  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51254 .x-view-drag-insert-above {
51255         border-top:1px dotted #3366cc;
51256 }
51257 .x-view-drag-insert-below {
51258         border-bottom:1px dotted #3366cc;
51259 }
51260 </code></pre>
51261  * 
51262  */
51263  
51264 Roo.DDView = function(container, tpl, config) {
51265     Roo.DDView.superclass.constructor.apply(this, arguments);
51266     this.getEl().setStyle("outline", "0px none");
51267     this.getEl().unselectable();
51268     if (this.dragGroup) {
51269                 this.setDraggable(this.dragGroup.split(","));
51270     }
51271     if (this.dropGroup) {
51272                 this.setDroppable(this.dropGroup.split(","));
51273     }
51274     if (this.deletable) {
51275         this.setDeletable();
51276     }
51277     this.isDirtyFlag = false;
51278         this.addEvents({
51279                 "drop" : true
51280         });
51281 };
51282
51283 Roo.extend(Roo.DDView, Roo.View, {
51284 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51285 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51286 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51287 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51288
51289         isFormField: true,
51290
51291         reset: Roo.emptyFn,
51292         
51293         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51294
51295         validate: function() {
51296                 return true;
51297         },
51298         
51299         destroy: function() {
51300                 this.purgeListeners();
51301                 this.getEl.removeAllListeners();
51302                 this.getEl().remove();
51303                 if (this.dragZone) {
51304                         if (this.dragZone.destroy) {
51305                                 this.dragZone.destroy();
51306                         }
51307                 }
51308                 if (this.dropZone) {
51309                         if (this.dropZone.destroy) {
51310                                 this.dropZone.destroy();
51311                         }
51312                 }
51313         },
51314
51315 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51316         getName: function() {
51317                 return this.name;
51318         },
51319
51320 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51321         setValue: function(v) {
51322                 if (!this.store) {
51323                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51324                 }
51325                 var data = {};
51326                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51327                 this.store.proxy = new Roo.data.MemoryProxy(data);
51328                 this.store.load();
51329         },
51330
51331 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51332         getValue: function() {
51333                 var result = '(';
51334                 this.store.each(function(rec) {
51335                         result += rec.id + ',';
51336                 });
51337                 return result.substr(0, result.length - 1) + ')';
51338         },
51339         
51340         getIds: function() {
51341                 var i = 0, result = new Array(this.store.getCount());
51342                 this.store.each(function(rec) {
51343                         result[i++] = rec.id;
51344                 });
51345                 return result;
51346         },
51347         
51348         isDirty: function() {
51349                 return this.isDirtyFlag;
51350         },
51351
51352 /**
51353  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51354  *      whole Element becomes the target, and this causes the drop gesture to append.
51355  */
51356     getTargetFromEvent : function(e) {
51357                 var target = e.getTarget();
51358                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51359                 target = target.parentNode;
51360                 }
51361                 if (!target) {
51362                         target = this.el.dom.lastChild || this.el.dom;
51363                 }
51364                 return target;
51365     },
51366
51367 /**
51368  *      Create the drag data which consists of an object which has the property "ddel" as
51369  *      the drag proxy element. 
51370  */
51371     getDragData : function(e) {
51372         var target = this.findItemFromChild(e.getTarget());
51373                 if(target) {
51374                         this.handleSelection(e);
51375                         var selNodes = this.getSelectedNodes();
51376             var dragData = {
51377                 source: this,
51378                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51379                 nodes: selNodes,
51380                 records: []
51381                         };
51382                         var selectedIndices = this.getSelectedIndexes();
51383                         for (var i = 0; i < selectedIndices.length; i++) {
51384                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51385                         }
51386                         if (selNodes.length == 1) {
51387                                 dragData.ddel = target.cloneNode(true); // the div element
51388                         } else {
51389                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51390                                 div.className = 'multi-proxy';
51391                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51392                                         div.appendChild(selNodes[i].cloneNode(true));
51393                                 }
51394                                 dragData.ddel = div;
51395                         }
51396             //console.log(dragData)
51397             //console.log(dragData.ddel.innerHTML)
51398                         return dragData;
51399                 }
51400         //console.log('nodragData')
51401                 return false;
51402     },
51403     
51404 /**     Specify to which ddGroup items in this DDView may be dragged. */
51405     setDraggable: function(ddGroup) {
51406         if (ddGroup instanceof Array) {
51407                 Roo.each(ddGroup, this.setDraggable, this);
51408                 return;
51409         }
51410         if (this.dragZone) {
51411                 this.dragZone.addToGroup(ddGroup);
51412         } else {
51413                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51414                                 containerScroll: true,
51415                                 ddGroup: ddGroup 
51416
51417                         });
51418 //                      Draggability implies selection. DragZone's mousedown selects the element.
51419                         if (!this.multiSelect) { this.singleSelect = true; }
51420
51421 //                      Wire the DragZone's handlers up to methods in *this*
51422                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51423                 }
51424     },
51425
51426 /**     Specify from which ddGroup this DDView accepts drops. */
51427     setDroppable: function(ddGroup) {
51428         if (ddGroup instanceof Array) {
51429                 Roo.each(ddGroup, this.setDroppable, this);
51430                 return;
51431         }
51432         if (this.dropZone) {
51433                 this.dropZone.addToGroup(ddGroup);
51434         } else {
51435                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51436                                 containerScroll: true,
51437                                 ddGroup: ddGroup
51438                         });
51439
51440 //                      Wire the DropZone's handlers up to methods in *this*
51441                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51442                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51443                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51444                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51445                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51446                 }
51447     },
51448
51449 /**     Decide whether to drop above or below a View node. */
51450     getDropPoint : function(e, n, dd){
51451         if (n == this.el.dom) { return "above"; }
51452                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51453                 var c = t + (b - t) / 2;
51454                 var y = Roo.lib.Event.getPageY(e);
51455                 if(y <= c) {
51456                         return "above";
51457                 }else{
51458                         return "below";
51459                 }
51460     },
51461
51462     onNodeEnter : function(n, dd, e, data){
51463                 return false;
51464     },
51465     
51466     onNodeOver : function(n, dd, e, data){
51467                 var pt = this.getDropPoint(e, n, dd);
51468                 // set the insert point style on the target node
51469                 var dragElClass = this.dropNotAllowed;
51470                 if (pt) {
51471                         var targetElClass;
51472                         if (pt == "above"){
51473                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51474                                 targetElClass = "x-view-drag-insert-above";
51475                         } else {
51476                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51477                                 targetElClass = "x-view-drag-insert-below";
51478                         }
51479                         if (this.lastInsertClass != targetElClass){
51480                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51481                                 this.lastInsertClass = targetElClass;
51482                         }
51483                 }
51484                 return dragElClass;
51485         },
51486
51487     onNodeOut : function(n, dd, e, data){
51488                 this.removeDropIndicators(n);
51489     },
51490
51491     onNodeDrop : function(n, dd, e, data){
51492         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51493                 return false;
51494         }
51495         var pt = this.getDropPoint(e, n, dd);
51496                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51497                 if (pt == "below") { insertAt++; }
51498                 for (var i = 0; i < data.records.length; i++) {
51499                         var r = data.records[i];
51500                         var dup = this.store.getById(r.id);
51501                         if (dup && (dd != this.dragZone)) {
51502                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51503                         } else {
51504                                 if (data.copy) {
51505                                         this.store.insert(insertAt++, r.copy());
51506                                 } else {
51507                                         data.source.isDirtyFlag = true;
51508                                         r.store.remove(r);
51509                                         this.store.insert(insertAt++, r);
51510                                 }
51511                                 this.isDirtyFlag = true;
51512                         }
51513                 }
51514                 this.dragZone.cachedTarget = null;
51515                 return true;
51516     },
51517
51518     removeDropIndicators : function(n){
51519                 if(n){
51520                         Roo.fly(n).removeClass([
51521                                 "x-view-drag-insert-above",
51522                                 "x-view-drag-insert-below"]);
51523                         this.lastInsertClass = "_noclass";
51524                 }
51525     },
51526
51527 /**
51528  *      Utility method. Add a delete option to the DDView's context menu.
51529  *      @param {String} imageUrl The URL of the "delete" icon image.
51530  */
51531         setDeletable: function(imageUrl) {
51532                 if (!this.singleSelect && !this.multiSelect) {
51533                         this.singleSelect = true;
51534                 }
51535                 var c = this.getContextMenu();
51536                 this.contextMenu.on("itemclick", function(item) {
51537                         switch (item.id) {
51538                                 case "delete":
51539                                         this.remove(this.getSelectedIndexes());
51540                                         break;
51541                         }
51542                 }, this);
51543                 this.contextMenu.add({
51544                         icon: imageUrl,
51545                         id: "delete",
51546                         text: 'Delete'
51547                 });
51548         },
51549         
51550 /**     Return the context menu for this DDView. */
51551         getContextMenu: function() {
51552                 if (!this.contextMenu) {
51553 //                      Create the View's context menu
51554                         this.contextMenu = new Roo.menu.Menu({
51555                                 id: this.id + "-contextmenu"
51556                         });
51557                         this.el.on("contextmenu", this.showContextMenu, this);
51558                 }
51559                 return this.contextMenu;
51560         },
51561         
51562         disableContextMenu: function() {
51563                 if (this.contextMenu) {
51564                         this.el.un("contextmenu", this.showContextMenu, this);
51565                 }
51566         },
51567
51568         showContextMenu: function(e, item) {
51569         item = this.findItemFromChild(e.getTarget());
51570                 if (item) {
51571                         e.stopEvent();
51572                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51573                         this.contextMenu.showAt(e.getXY());
51574             }
51575     },
51576
51577 /**
51578  *      Remove {@link Roo.data.Record}s at the specified indices.
51579  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51580  */
51581     remove: function(selectedIndices) {
51582                 selectedIndices = [].concat(selectedIndices);
51583                 for (var i = 0; i < selectedIndices.length; i++) {
51584                         var rec = this.store.getAt(selectedIndices[i]);
51585                         this.store.remove(rec);
51586                 }
51587     },
51588
51589 /**
51590  *      Double click fires the event, but also, if this is draggable, and there is only one other
51591  *      related DropZone, it transfers the selected node.
51592  */
51593     onDblClick : function(e){
51594         var item = this.findItemFromChild(e.getTarget());
51595         if(item){
51596             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51597                 return false;
51598             }
51599             if (this.dragGroup) {
51600                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51601                     while (targets.indexOf(this.dropZone) > -1) {
51602                             targets.remove(this.dropZone);
51603                                 }
51604                     if (targets.length == 1) {
51605                                         this.dragZone.cachedTarget = null;
51606                         var el = Roo.get(targets[0].getEl());
51607                         var box = el.getBox(true);
51608                         targets[0].onNodeDrop(el.dom, {
51609                                 target: el.dom,
51610                                 xy: [box.x, box.y + box.height - 1]
51611                         }, null, this.getDragData(e));
51612                     }
51613                 }
51614         }
51615     },
51616     
51617     handleSelection: function(e) {
51618                 this.dragZone.cachedTarget = null;
51619         var item = this.findItemFromChild(e.getTarget());
51620         if (!item) {
51621                 this.clearSelections(true);
51622                 return;
51623         }
51624                 if (item && (this.multiSelect || this.singleSelect)){
51625                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51626                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51627                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51628                                 this.unselect(item);
51629                         } else {
51630                                 this.select(item, this.multiSelect && e.ctrlKey);
51631                                 this.lastSelection = item;
51632                         }
51633                 }
51634     },
51635
51636     onItemClick : function(item, index, e){
51637                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51638                         return false;
51639                 }
51640                 return true;
51641     },
51642
51643     unselect : function(nodeInfo, suppressEvent){
51644                 var node = this.getNode(nodeInfo);
51645                 if(node && this.isSelected(node)){
51646                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51647                                 Roo.fly(node).removeClass(this.selectedClass);
51648                                 this.selections.remove(node);
51649                                 if(!suppressEvent){
51650                                         this.fireEvent("selectionchange", this, this.selections);
51651                                 }
51652                         }
51653                 }
51654     }
51655 });
51656 /*
51657  * Based on:
51658  * Ext JS Library 1.1.1
51659  * Copyright(c) 2006-2007, Ext JS, LLC.
51660  *
51661  * Originally Released Under LGPL - original licence link has changed is not relivant.
51662  *
51663  * Fork - LGPL
51664  * <script type="text/javascript">
51665  */
51666  
51667 /**
51668  * @class Roo.LayoutManager
51669  * @extends Roo.util.Observable
51670  * Base class for layout managers.
51671  */
51672 Roo.LayoutManager = function(container, config){
51673     Roo.LayoutManager.superclass.constructor.call(this);
51674     this.el = Roo.get(container);
51675     // ie scrollbar fix
51676     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51677         document.body.scroll = "no";
51678     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51679         this.el.position('relative');
51680     }
51681     this.id = this.el.id;
51682     this.el.addClass("x-layout-container");
51683     /** false to disable window resize monitoring @type Boolean */
51684     this.monitorWindowResize = true;
51685     this.regions = {};
51686     this.addEvents({
51687         /**
51688          * @event layout
51689          * Fires when a layout is performed. 
51690          * @param {Roo.LayoutManager} this
51691          */
51692         "layout" : true,
51693         /**
51694          * @event regionresized
51695          * Fires when the user resizes a region. 
51696          * @param {Roo.LayoutRegion} region The resized region
51697          * @param {Number} newSize The new size (width for east/west, height for north/south)
51698          */
51699         "regionresized" : true,
51700         /**
51701          * @event regioncollapsed
51702          * Fires when a region is collapsed. 
51703          * @param {Roo.LayoutRegion} region The collapsed region
51704          */
51705         "regioncollapsed" : true,
51706         /**
51707          * @event regionexpanded
51708          * Fires when a region is expanded.  
51709          * @param {Roo.LayoutRegion} region The expanded region
51710          */
51711         "regionexpanded" : true
51712     });
51713     this.updating = false;
51714     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51715 };
51716
51717 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51718     /**
51719      * Returns true if this layout is currently being updated
51720      * @return {Boolean}
51721      */
51722     isUpdating : function(){
51723         return this.updating; 
51724     },
51725     
51726     /**
51727      * Suspend the LayoutManager from doing auto-layouts while
51728      * making multiple add or remove calls
51729      */
51730     beginUpdate : function(){
51731         this.updating = true;    
51732     },
51733     
51734     /**
51735      * Restore auto-layouts and optionally disable the manager from performing a layout
51736      * @param {Boolean} noLayout true to disable a layout update 
51737      */
51738     endUpdate : function(noLayout){
51739         this.updating = false;
51740         if(!noLayout){
51741             this.layout();
51742         }    
51743     },
51744     
51745     layout: function(){
51746         
51747     },
51748     
51749     onRegionResized : function(region, newSize){
51750         this.fireEvent("regionresized", region, newSize);
51751         this.layout();
51752     },
51753     
51754     onRegionCollapsed : function(region){
51755         this.fireEvent("regioncollapsed", region);
51756     },
51757     
51758     onRegionExpanded : function(region){
51759         this.fireEvent("regionexpanded", region);
51760     },
51761         
51762     /**
51763      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51764      * performs box-model adjustments.
51765      * @return {Object} The size as an object {width: (the width), height: (the height)}
51766      */
51767     getViewSize : function(){
51768         var size;
51769         if(this.el.dom != document.body){
51770             size = this.el.getSize();
51771         }else{
51772             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51773         }
51774         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51775         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51776         return size;
51777     },
51778     
51779     /**
51780      * Returns the Element this layout is bound to.
51781      * @return {Roo.Element}
51782      */
51783     getEl : function(){
51784         return this.el;
51785     },
51786     
51787     /**
51788      * Returns the specified region.
51789      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51790      * @return {Roo.LayoutRegion}
51791      */
51792     getRegion : function(target){
51793         return this.regions[target.toLowerCase()];
51794     },
51795     
51796     onWindowResize : function(){
51797         if(this.monitorWindowResize){
51798             this.layout();
51799         }
51800     }
51801 });/*
51802  * Based on:
51803  * Ext JS Library 1.1.1
51804  * Copyright(c) 2006-2007, Ext JS, LLC.
51805  *
51806  * Originally Released Under LGPL - original licence link has changed is not relivant.
51807  *
51808  * Fork - LGPL
51809  * <script type="text/javascript">
51810  */
51811 /**
51812  * @class Roo.BorderLayout
51813  * @extends Roo.LayoutManager
51814  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51815  * please see: <br><br>
51816  * <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>
51817  * <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>
51818  * Example:
51819  <pre><code>
51820  var layout = new Roo.BorderLayout(document.body, {
51821     north: {
51822         initialSize: 25,
51823         titlebar: false
51824     },
51825     west: {
51826         split:true,
51827         initialSize: 200,
51828         minSize: 175,
51829         maxSize: 400,
51830         titlebar: true,
51831         collapsible: true
51832     },
51833     east: {
51834         split:true,
51835         initialSize: 202,
51836         minSize: 175,
51837         maxSize: 400,
51838         titlebar: true,
51839         collapsible: true
51840     },
51841     south: {
51842         split:true,
51843         initialSize: 100,
51844         minSize: 100,
51845         maxSize: 200,
51846         titlebar: true,
51847         collapsible: true
51848     },
51849     center: {
51850         titlebar: true,
51851         autoScroll:true,
51852         resizeTabs: true,
51853         minTabWidth: 50,
51854         preferredTabWidth: 150
51855     }
51856 });
51857
51858 // shorthand
51859 var CP = Roo.ContentPanel;
51860
51861 layout.beginUpdate();
51862 layout.add("north", new CP("north", "North"));
51863 layout.add("south", new CP("south", {title: "South", closable: true}));
51864 layout.add("west", new CP("west", {title: "West"}));
51865 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51866 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51867 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51868 layout.getRegion("center").showPanel("center1");
51869 layout.endUpdate();
51870 </code></pre>
51871
51872 <b>The container the layout is rendered into can be either the body element or any other element.
51873 If it is not the body element, the container needs to either be an absolute positioned element,
51874 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51875 the container size if it is not the body element.</b>
51876
51877 * @constructor
51878 * Create a new BorderLayout
51879 * @param {String/HTMLElement/Element} container The container this layout is bound to
51880 * @param {Object} config Configuration options
51881  */
51882 Roo.BorderLayout = function(container, config){
51883     config = config || {};
51884     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51885     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51886     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51887         var target = this.factory.validRegions[i];
51888         if(config[target]){
51889             this.addRegion(target, config[target]);
51890         }
51891     }
51892 };
51893
51894 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51895     /**
51896      * Creates and adds a new region if it doesn't already exist.
51897      * @param {String} target The target region key (north, south, east, west or center).
51898      * @param {Object} config The regions config object
51899      * @return {BorderLayoutRegion} The new region
51900      */
51901     addRegion : function(target, config){
51902         if(!this.regions[target]){
51903             var r = this.factory.create(target, this, config);
51904             this.bindRegion(target, r);
51905         }
51906         return this.regions[target];
51907     },
51908
51909     // private (kinda)
51910     bindRegion : function(name, r){
51911         this.regions[name] = r;
51912         r.on("visibilitychange", this.layout, this);
51913         r.on("paneladded", this.layout, this);
51914         r.on("panelremoved", this.layout, this);
51915         r.on("invalidated", this.layout, this);
51916         r.on("resized", this.onRegionResized, this);
51917         r.on("collapsed", this.onRegionCollapsed, this);
51918         r.on("expanded", this.onRegionExpanded, this);
51919     },
51920
51921     /**
51922      * Performs a layout update.
51923      */
51924     layout : function(){
51925         if(this.updating) {
51926             return;
51927         }
51928         var size = this.getViewSize();
51929         var w = size.width;
51930         var h = size.height;
51931         var centerW = w;
51932         var centerH = h;
51933         var centerY = 0;
51934         var centerX = 0;
51935         //var x = 0, y = 0;
51936
51937         var rs = this.regions;
51938         var north = rs["north"];
51939         var south = rs["south"]; 
51940         var west = rs["west"];
51941         var east = rs["east"];
51942         var center = rs["center"];
51943         //if(this.hideOnLayout){ // not supported anymore
51944             //c.el.setStyle("display", "none");
51945         //}
51946         if(north && north.isVisible()){
51947             var b = north.getBox();
51948             var m = north.getMargins();
51949             b.width = w - (m.left+m.right);
51950             b.x = m.left;
51951             b.y = m.top;
51952             centerY = b.height + b.y + m.bottom;
51953             centerH -= centerY;
51954             north.updateBox(this.safeBox(b));
51955         }
51956         if(south && south.isVisible()){
51957             var b = south.getBox();
51958             var m = south.getMargins();
51959             b.width = w - (m.left+m.right);
51960             b.x = m.left;
51961             var totalHeight = (b.height + m.top + m.bottom);
51962             b.y = h - totalHeight + m.top;
51963             centerH -= totalHeight;
51964             south.updateBox(this.safeBox(b));
51965         }
51966         if(west && west.isVisible()){
51967             var b = west.getBox();
51968             var m = west.getMargins();
51969             b.height = centerH - (m.top+m.bottom);
51970             b.x = m.left;
51971             b.y = centerY + m.top;
51972             var totalWidth = (b.width + m.left + m.right);
51973             centerX += totalWidth;
51974             centerW -= totalWidth;
51975             west.updateBox(this.safeBox(b));
51976         }
51977         if(east && east.isVisible()){
51978             var b = east.getBox();
51979             var m = east.getMargins();
51980             b.height = centerH - (m.top+m.bottom);
51981             var totalWidth = (b.width + m.left + m.right);
51982             b.x = w - totalWidth + m.left;
51983             b.y = centerY + m.top;
51984             centerW -= totalWidth;
51985             east.updateBox(this.safeBox(b));
51986         }
51987         if(center){
51988             var m = center.getMargins();
51989             var centerBox = {
51990                 x: centerX + m.left,
51991                 y: centerY + m.top,
51992                 width: centerW - (m.left+m.right),
51993                 height: centerH - (m.top+m.bottom)
51994             };
51995             //if(this.hideOnLayout){
51996                 //center.el.setStyle("display", "block");
51997             //}
51998             center.updateBox(this.safeBox(centerBox));
51999         }
52000         this.el.repaint();
52001         this.fireEvent("layout", this);
52002     },
52003
52004     // private
52005     safeBox : function(box){
52006         box.width = Math.max(0, box.width);
52007         box.height = Math.max(0, box.height);
52008         return box;
52009     },
52010
52011     /**
52012      * Adds a ContentPanel (or subclass) to this layout.
52013      * @param {String} target The target region key (north, south, east, west or center).
52014      * @param {Roo.ContentPanel} panel The panel to add
52015      * @return {Roo.ContentPanel} The added panel
52016      */
52017     add : function(target, panel){
52018          
52019         target = target.toLowerCase();
52020         return this.regions[target].add(panel);
52021     },
52022
52023     /**
52024      * Remove a ContentPanel (or subclass) to this layout.
52025      * @param {String} target The target region key (north, south, east, west or center).
52026      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52027      * @return {Roo.ContentPanel} The removed panel
52028      */
52029     remove : function(target, panel){
52030         target = target.toLowerCase();
52031         return this.regions[target].remove(panel);
52032     },
52033
52034     /**
52035      * Searches all regions for a panel with the specified id
52036      * @param {String} panelId
52037      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52038      */
52039     findPanel : function(panelId){
52040         var rs = this.regions;
52041         for(var target in rs){
52042             if(typeof rs[target] != "function"){
52043                 var p = rs[target].getPanel(panelId);
52044                 if(p){
52045                     return p;
52046                 }
52047             }
52048         }
52049         return null;
52050     },
52051
52052     /**
52053      * Searches all regions for a panel with the specified id and activates (shows) it.
52054      * @param {String/ContentPanel} panelId The panels id or the panel itself
52055      * @return {Roo.ContentPanel} The shown panel or null
52056      */
52057     showPanel : function(panelId) {
52058       var rs = this.regions;
52059       for(var target in rs){
52060          var r = rs[target];
52061          if(typeof r != "function"){
52062             if(r.hasPanel(panelId)){
52063                return r.showPanel(panelId);
52064             }
52065          }
52066       }
52067       return null;
52068    },
52069
52070    /**
52071      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52072      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52073      */
52074     restoreState : function(provider){
52075         if(!provider){
52076             provider = Roo.state.Manager;
52077         }
52078         var sm = new Roo.LayoutStateManager();
52079         sm.init(this, provider);
52080     },
52081
52082     /**
52083      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52084      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52085      * a valid ContentPanel config object.  Example:
52086      * <pre><code>
52087 // Create the main layout
52088 var layout = new Roo.BorderLayout('main-ct', {
52089     west: {
52090         split:true,
52091         minSize: 175,
52092         titlebar: true
52093     },
52094     center: {
52095         title:'Components'
52096     }
52097 }, 'main-ct');
52098
52099 // Create and add multiple ContentPanels at once via configs
52100 layout.batchAdd({
52101    west: {
52102        id: 'source-files',
52103        autoCreate:true,
52104        title:'Ext Source Files',
52105        autoScroll:true,
52106        fitToFrame:true
52107    },
52108    center : {
52109        el: cview,
52110        autoScroll:true,
52111        fitToFrame:true,
52112        toolbar: tb,
52113        resizeEl:'cbody'
52114    }
52115 });
52116 </code></pre>
52117      * @param {Object} regions An object containing ContentPanel configs by region name
52118      */
52119     batchAdd : function(regions){
52120         this.beginUpdate();
52121         for(var rname in regions){
52122             var lr = this.regions[rname];
52123             if(lr){
52124                 this.addTypedPanels(lr, regions[rname]);
52125             }
52126         }
52127         this.endUpdate();
52128     },
52129
52130     // private
52131     addTypedPanels : function(lr, ps){
52132         if(typeof ps == 'string'){
52133             lr.add(new Roo.ContentPanel(ps));
52134         }
52135         else if(ps instanceof Array){
52136             for(var i =0, len = ps.length; i < len; i++){
52137                 this.addTypedPanels(lr, ps[i]);
52138             }
52139         }
52140         else if(!ps.events){ // raw config?
52141             var el = ps.el;
52142             delete ps.el; // prevent conflict
52143             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52144         }
52145         else {  // panel object assumed!
52146             lr.add(ps);
52147         }
52148     },
52149     /**
52150      * Adds a xtype elements to the layout.
52151      * <pre><code>
52152
52153 layout.addxtype({
52154        xtype : 'ContentPanel',
52155        region: 'west',
52156        items: [ .... ]
52157    }
52158 );
52159
52160 layout.addxtype({
52161         xtype : 'NestedLayoutPanel',
52162         region: 'west',
52163         layout: {
52164            center: { },
52165            west: { }   
52166         },
52167         items : [ ... list of content panels or nested layout panels.. ]
52168    }
52169 );
52170 </code></pre>
52171      * @param {Object} cfg Xtype definition of item to add.
52172      */
52173     addxtype : function(cfg)
52174     {
52175         // basically accepts a pannel...
52176         // can accept a layout region..!?!?
52177         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52178         
52179         if (!cfg.xtype.match(/Panel$/)) {
52180             return false;
52181         }
52182         var ret = false;
52183         
52184         if (typeof(cfg.region) == 'undefined') {
52185             Roo.log("Failed to add Panel, region was not set");
52186             Roo.log(cfg);
52187             return false;
52188         }
52189         var region = cfg.region;
52190         delete cfg.region;
52191         
52192           
52193         var xitems = [];
52194         if (cfg.items) {
52195             xitems = cfg.items;
52196             delete cfg.items;
52197         }
52198         var nb = false;
52199         
52200         switch(cfg.xtype) 
52201         {
52202             case 'ContentPanel':  // ContentPanel (el, cfg)
52203             case 'ScrollPanel':  // ContentPanel (el, cfg)
52204             case 'ViewPanel': 
52205                 if(cfg.autoCreate) {
52206                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52207                 } else {
52208                     var el = this.el.createChild();
52209                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52210                 }
52211                 
52212                 this.add(region, ret);
52213                 break;
52214             
52215             
52216             case 'TreePanel': // our new panel!
52217                 cfg.el = this.el.createChild();
52218                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52219                 this.add(region, ret);
52220                 break;
52221             
52222             case 'NestedLayoutPanel': 
52223                 // create a new Layout (which is  a Border Layout...
52224                 var el = this.el.createChild();
52225                 var clayout = cfg.layout;
52226                 delete cfg.layout;
52227                 clayout.items   = clayout.items  || [];
52228                 // replace this exitems with the clayout ones..
52229                 xitems = clayout.items;
52230                  
52231                 
52232                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52233                     cfg.background = false;
52234                 }
52235                 var layout = new Roo.BorderLayout(el, clayout);
52236                 
52237                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52238                 //console.log('adding nested layout panel '  + cfg.toSource());
52239                 this.add(region, ret);
52240                 nb = {}; /// find first...
52241                 break;
52242                 
52243             case 'GridPanel': 
52244             
52245                 // needs grid and region
52246                 
52247                 //var el = this.getRegion(region).el.createChild();
52248                 var el = this.el.createChild();
52249                 // create the grid first...
52250                 
52251                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52252                 delete cfg.grid;
52253                 if (region == 'center' && this.active ) {
52254                     cfg.background = false;
52255                 }
52256                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52257                 
52258                 this.add(region, ret);
52259                 if (cfg.background) {
52260                     ret.on('activate', function(gp) {
52261                         if (!gp.grid.rendered) {
52262                             gp.grid.render();
52263                         }
52264                     });
52265                 } else {
52266                     grid.render();
52267                 }
52268                 break;
52269            
52270            
52271            
52272                 
52273                 
52274                 
52275             default:
52276                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52277                     
52278                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52279                     this.add(region, ret);
52280                 } else {
52281                 
52282                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52283                     return null;
52284                 }
52285                 
52286              // GridPanel (grid, cfg)
52287             
52288         }
52289         this.beginUpdate();
52290         // add children..
52291         var region = '';
52292         var abn = {};
52293         Roo.each(xitems, function(i)  {
52294             region = nb && i.region ? i.region : false;
52295             
52296             var add = ret.addxtype(i);
52297            
52298             if (region) {
52299                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52300                 if (!i.background) {
52301                     abn[region] = nb[region] ;
52302                 }
52303             }
52304             
52305         });
52306         this.endUpdate();
52307
52308         // make the last non-background panel active..
52309         //if (nb) { Roo.log(abn); }
52310         if (nb) {
52311             
52312             for(var r in abn) {
52313                 region = this.getRegion(r);
52314                 if (region) {
52315                     // tried using nb[r], but it does not work..
52316                      
52317                     region.showPanel(abn[r]);
52318                    
52319                 }
52320             }
52321         }
52322         return ret;
52323         
52324     }
52325 });
52326
52327 /**
52328  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52329  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52330  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52331  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52332  * <pre><code>
52333 // shorthand
52334 var CP = Roo.ContentPanel;
52335
52336 var layout = Roo.BorderLayout.create({
52337     north: {
52338         initialSize: 25,
52339         titlebar: false,
52340         panels: [new CP("north", "North")]
52341     },
52342     west: {
52343         split:true,
52344         initialSize: 200,
52345         minSize: 175,
52346         maxSize: 400,
52347         titlebar: true,
52348         collapsible: true,
52349         panels: [new CP("west", {title: "West"})]
52350     },
52351     east: {
52352         split:true,
52353         initialSize: 202,
52354         minSize: 175,
52355         maxSize: 400,
52356         titlebar: true,
52357         collapsible: true,
52358         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52359     },
52360     south: {
52361         split:true,
52362         initialSize: 100,
52363         minSize: 100,
52364         maxSize: 200,
52365         titlebar: true,
52366         collapsible: true,
52367         panels: [new CP("south", {title: "South", closable: true})]
52368     },
52369     center: {
52370         titlebar: true,
52371         autoScroll:true,
52372         resizeTabs: true,
52373         minTabWidth: 50,
52374         preferredTabWidth: 150,
52375         panels: [
52376             new CP("center1", {title: "Close Me", closable: true}),
52377             new CP("center2", {title: "Center Panel", closable: false})
52378         ]
52379     }
52380 }, document.body);
52381
52382 layout.getRegion("center").showPanel("center1");
52383 </code></pre>
52384  * @param config
52385  * @param targetEl
52386  */
52387 Roo.BorderLayout.create = function(config, targetEl){
52388     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52389     layout.beginUpdate();
52390     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52391     for(var j = 0, jlen = regions.length; j < jlen; j++){
52392         var lr = regions[j];
52393         if(layout.regions[lr] && config[lr].panels){
52394             var r = layout.regions[lr];
52395             var ps = config[lr].panels;
52396             layout.addTypedPanels(r, ps);
52397         }
52398     }
52399     layout.endUpdate();
52400     return layout;
52401 };
52402
52403 // private
52404 Roo.BorderLayout.RegionFactory = {
52405     // private
52406     validRegions : ["north","south","east","west","center"],
52407
52408     // private
52409     create : function(target, mgr, config){
52410         target = target.toLowerCase();
52411         if(config.lightweight || config.basic){
52412             return new Roo.BasicLayoutRegion(mgr, config, target);
52413         }
52414         switch(target){
52415             case "north":
52416                 return new Roo.NorthLayoutRegion(mgr, config);
52417             case "south":
52418                 return new Roo.SouthLayoutRegion(mgr, config);
52419             case "east":
52420                 return new Roo.EastLayoutRegion(mgr, config);
52421             case "west":
52422                 return new Roo.WestLayoutRegion(mgr, config);
52423             case "center":
52424                 return new Roo.CenterLayoutRegion(mgr, config);
52425         }
52426         throw 'Layout region "'+target+'" not supported.';
52427     }
52428 };/*
52429  * Based on:
52430  * Ext JS Library 1.1.1
52431  * Copyright(c) 2006-2007, Ext JS, LLC.
52432  *
52433  * Originally Released Under LGPL - original licence link has changed is not relivant.
52434  *
52435  * Fork - LGPL
52436  * <script type="text/javascript">
52437  */
52438  
52439 /**
52440  * @class Roo.BasicLayoutRegion
52441  * @extends Roo.util.Observable
52442  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52443  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52444  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52445  */
52446 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52447     this.mgr = mgr;
52448     this.position  = pos;
52449     this.events = {
52450         /**
52451          * @scope Roo.BasicLayoutRegion
52452          */
52453         
52454         /**
52455          * @event beforeremove
52456          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52457          * @param {Roo.LayoutRegion} this
52458          * @param {Roo.ContentPanel} panel The panel
52459          * @param {Object} e The cancel event object
52460          */
52461         "beforeremove" : true,
52462         /**
52463          * @event invalidated
52464          * Fires when the layout for this region is changed.
52465          * @param {Roo.LayoutRegion} this
52466          */
52467         "invalidated" : true,
52468         /**
52469          * @event visibilitychange
52470          * Fires when this region is shown or hidden 
52471          * @param {Roo.LayoutRegion} this
52472          * @param {Boolean} visibility true or false
52473          */
52474         "visibilitychange" : true,
52475         /**
52476          * @event paneladded
52477          * Fires when a panel is added. 
52478          * @param {Roo.LayoutRegion} this
52479          * @param {Roo.ContentPanel} panel The panel
52480          */
52481         "paneladded" : true,
52482         /**
52483          * @event panelremoved
52484          * Fires when a panel is removed. 
52485          * @param {Roo.LayoutRegion} this
52486          * @param {Roo.ContentPanel} panel The panel
52487          */
52488         "panelremoved" : true,
52489         /**
52490          * @event beforecollapse
52491          * Fires when this region before collapse.
52492          * @param {Roo.LayoutRegion} this
52493          */
52494         "beforecollapse" : true,
52495         /**
52496          * @event collapsed
52497          * Fires when this region is collapsed.
52498          * @param {Roo.LayoutRegion} this
52499          */
52500         "collapsed" : true,
52501         /**
52502          * @event expanded
52503          * Fires when this region is expanded.
52504          * @param {Roo.LayoutRegion} this
52505          */
52506         "expanded" : true,
52507         /**
52508          * @event slideshow
52509          * Fires when this region is slid into view.
52510          * @param {Roo.LayoutRegion} this
52511          */
52512         "slideshow" : true,
52513         /**
52514          * @event slidehide
52515          * Fires when this region slides out of view. 
52516          * @param {Roo.LayoutRegion} this
52517          */
52518         "slidehide" : true,
52519         /**
52520          * @event panelactivated
52521          * Fires when a panel is activated. 
52522          * @param {Roo.LayoutRegion} this
52523          * @param {Roo.ContentPanel} panel The activated panel
52524          */
52525         "panelactivated" : true,
52526         /**
52527          * @event resized
52528          * Fires when the user resizes this region. 
52529          * @param {Roo.LayoutRegion} this
52530          * @param {Number} newSize The new size (width for east/west, height for north/south)
52531          */
52532         "resized" : true
52533     };
52534     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52535     this.panels = new Roo.util.MixedCollection();
52536     this.panels.getKey = this.getPanelId.createDelegate(this);
52537     this.box = null;
52538     this.activePanel = null;
52539     // ensure listeners are added...
52540     
52541     if (config.listeners || config.events) {
52542         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52543             listeners : config.listeners || {},
52544             events : config.events || {}
52545         });
52546     }
52547     
52548     if(skipConfig !== true){
52549         this.applyConfig(config);
52550     }
52551 };
52552
52553 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52554     getPanelId : function(p){
52555         return p.getId();
52556     },
52557     
52558     applyConfig : function(config){
52559         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52560         this.config = config;
52561         
52562     },
52563     
52564     /**
52565      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52566      * the width, for horizontal (north, south) the height.
52567      * @param {Number} newSize The new width or height
52568      */
52569     resizeTo : function(newSize){
52570         var el = this.el ? this.el :
52571                  (this.activePanel ? this.activePanel.getEl() : null);
52572         if(el){
52573             switch(this.position){
52574                 case "east":
52575                 case "west":
52576                     el.setWidth(newSize);
52577                     this.fireEvent("resized", this, newSize);
52578                 break;
52579                 case "north":
52580                 case "south":
52581                     el.setHeight(newSize);
52582                     this.fireEvent("resized", this, newSize);
52583                 break;                
52584             }
52585         }
52586     },
52587     
52588     getBox : function(){
52589         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52590     },
52591     
52592     getMargins : function(){
52593         return this.margins;
52594     },
52595     
52596     updateBox : function(box){
52597         this.box = box;
52598         var el = this.activePanel.getEl();
52599         el.dom.style.left = box.x + "px";
52600         el.dom.style.top = box.y + "px";
52601         this.activePanel.setSize(box.width, box.height);
52602     },
52603     
52604     /**
52605      * Returns the container element for this region.
52606      * @return {Roo.Element}
52607      */
52608     getEl : function(){
52609         return this.activePanel;
52610     },
52611     
52612     /**
52613      * Returns true if this region is currently visible.
52614      * @return {Boolean}
52615      */
52616     isVisible : function(){
52617         return this.activePanel ? true : false;
52618     },
52619     
52620     setActivePanel : function(panel){
52621         panel = this.getPanel(panel);
52622         if(this.activePanel && this.activePanel != panel){
52623             this.activePanel.setActiveState(false);
52624             this.activePanel.getEl().setLeftTop(-10000,-10000);
52625         }
52626         this.activePanel = panel;
52627         panel.setActiveState(true);
52628         if(this.box){
52629             panel.setSize(this.box.width, this.box.height);
52630         }
52631         this.fireEvent("panelactivated", this, panel);
52632         this.fireEvent("invalidated");
52633     },
52634     
52635     /**
52636      * Show the specified panel.
52637      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52638      * @return {Roo.ContentPanel} The shown panel or null
52639      */
52640     showPanel : function(panel){
52641         if(panel = this.getPanel(panel)){
52642             this.setActivePanel(panel);
52643         }
52644         return panel;
52645     },
52646     
52647     /**
52648      * Get the active panel for this region.
52649      * @return {Roo.ContentPanel} The active panel or null
52650      */
52651     getActivePanel : function(){
52652         return this.activePanel;
52653     },
52654     
52655     /**
52656      * Add the passed ContentPanel(s)
52657      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52658      * @return {Roo.ContentPanel} The panel added (if only one was added)
52659      */
52660     add : function(panel){
52661         if(arguments.length > 1){
52662             for(var i = 0, len = arguments.length; i < len; i++) {
52663                 this.add(arguments[i]);
52664             }
52665             return null;
52666         }
52667         if(this.hasPanel(panel)){
52668             this.showPanel(panel);
52669             return panel;
52670         }
52671         var el = panel.getEl();
52672         if(el.dom.parentNode != this.mgr.el.dom){
52673             this.mgr.el.dom.appendChild(el.dom);
52674         }
52675         if(panel.setRegion){
52676             panel.setRegion(this);
52677         }
52678         this.panels.add(panel);
52679         el.setStyle("position", "absolute");
52680         if(!panel.background){
52681             this.setActivePanel(panel);
52682             if(this.config.initialSize && this.panels.getCount()==1){
52683                 this.resizeTo(this.config.initialSize);
52684             }
52685         }
52686         this.fireEvent("paneladded", this, panel);
52687         return panel;
52688     },
52689     
52690     /**
52691      * Returns true if the panel is in this region.
52692      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52693      * @return {Boolean}
52694      */
52695     hasPanel : function(panel){
52696         if(typeof panel == "object"){ // must be panel obj
52697             panel = panel.getId();
52698         }
52699         return this.getPanel(panel) ? true : false;
52700     },
52701     
52702     /**
52703      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52704      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52705      * @param {Boolean} preservePanel Overrides the config preservePanel option
52706      * @return {Roo.ContentPanel} The panel that was removed
52707      */
52708     remove : function(panel, preservePanel){
52709         panel = this.getPanel(panel);
52710         if(!panel){
52711             return null;
52712         }
52713         var e = {};
52714         this.fireEvent("beforeremove", this, panel, e);
52715         if(e.cancel === true){
52716             return null;
52717         }
52718         var panelId = panel.getId();
52719         this.panels.removeKey(panelId);
52720         return panel;
52721     },
52722     
52723     /**
52724      * Returns the panel specified or null if it's not in this region.
52725      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52726      * @return {Roo.ContentPanel}
52727      */
52728     getPanel : function(id){
52729         if(typeof id == "object"){ // must be panel obj
52730             return id;
52731         }
52732         return this.panels.get(id);
52733     },
52734     
52735     /**
52736      * Returns this regions position (north/south/east/west/center).
52737      * @return {String} 
52738      */
52739     getPosition: function(){
52740         return this.position;    
52741     }
52742 });/*
52743  * Based on:
52744  * Ext JS Library 1.1.1
52745  * Copyright(c) 2006-2007, Ext JS, LLC.
52746  *
52747  * Originally Released Under LGPL - original licence link has changed is not relivant.
52748  *
52749  * Fork - LGPL
52750  * <script type="text/javascript">
52751  */
52752  
52753 /**
52754  * @class Roo.LayoutRegion
52755  * @extends Roo.BasicLayoutRegion
52756  * This class represents a region in a layout manager.
52757  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52758  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52759  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52760  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52761  * @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})
52762  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52763  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52764  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52765  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52766  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52767  * @cfg {String}    title           The title for the region (overrides panel titles)
52768  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52769  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52770  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52771  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52772  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52773  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52774  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52775  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52776  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52777  * @cfg {Boolean}   showPin         True to show a pin button
52778  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52779  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52780  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52781  * @cfg {Number}    width           For East/West panels
52782  * @cfg {Number}    height          For North/South panels
52783  * @cfg {Boolean}   split           To show the splitter
52784  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52785  */
52786 Roo.LayoutRegion = function(mgr, config, pos){
52787     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52788     var dh = Roo.DomHelper;
52789     /** This region's container element 
52790     * @type Roo.Element */
52791     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52792     /** This region's title element 
52793     * @type Roo.Element */
52794
52795     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52796         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52797         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52798     ]}, true);
52799     this.titleEl.enableDisplayMode();
52800     /** This region's title text element 
52801     * @type HTMLElement */
52802     this.titleTextEl = this.titleEl.dom.firstChild;
52803     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52804     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52805     this.closeBtn.enableDisplayMode();
52806     this.closeBtn.on("click", this.closeClicked, this);
52807     this.closeBtn.hide();
52808
52809     this.createBody(config);
52810     this.visible = true;
52811     this.collapsed = false;
52812
52813     if(config.hideWhenEmpty){
52814         this.hide();
52815         this.on("paneladded", this.validateVisibility, this);
52816         this.on("panelremoved", this.validateVisibility, this);
52817     }
52818     this.applyConfig(config);
52819 };
52820
52821 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52822
52823     createBody : function(){
52824         /** This region's body element 
52825         * @type Roo.Element */
52826         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52827     },
52828
52829     applyConfig : function(c){
52830         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52831             var dh = Roo.DomHelper;
52832             if(c.titlebar !== false){
52833                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52834                 this.collapseBtn.on("click", this.collapse, this);
52835                 this.collapseBtn.enableDisplayMode();
52836
52837                 if(c.showPin === true || this.showPin){
52838                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52839                     this.stickBtn.enableDisplayMode();
52840                     this.stickBtn.on("click", this.expand, this);
52841                     this.stickBtn.hide();
52842                 }
52843             }
52844             /** This region's collapsed element
52845             * @type Roo.Element */
52846             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52847                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52848             ]}, true);
52849             if(c.floatable !== false){
52850                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52851                this.collapsedEl.on("click", this.collapseClick, this);
52852             }
52853
52854             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52855                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52856                    id: "message", unselectable: "on", style:{"float":"left"}});
52857                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52858              }
52859             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52860             this.expandBtn.on("click", this.expand, this);
52861         }
52862         if(this.collapseBtn){
52863             this.collapseBtn.setVisible(c.collapsible == true);
52864         }
52865         this.cmargins = c.cmargins || this.cmargins ||
52866                          (this.position == "west" || this.position == "east" ?
52867                              {top: 0, left: 2, right:2, bottom: 0} :
52868                              {top: 2, left: 0, right:0, bottom: 2});
52869         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52870         this.bottomTabs = c.tabPosition != "top";
52871         this.autoScroll = c.autoScroll || false;
52872         if(this.autoScroll){
52873             this.bodyEl.setStyle("overflow", "auto");
52874         }else{
52875             this.bodyEl.setStyle("overflow", "hidden");
52876         }
52877         //if(c.titlebar !== false){
52878             if((!c.titlebar && !c.title) || c.titlebar === false){
52879                 this.titleEl.hide();
52880             }else{
52881                 this.titleEl.show();
52882                 if(c.title){
52883                     this.titleTextEl.innerHTML = c.title;
52884                 }
52885             }
52886         //}
52887         this.duration = c.duration || .30;
52888         this.slideDuration = c.slideDuration || .45;
52889         this.config = c;
52890         if(c.collapsed){
52891             this.collapse(true);
52892         }
52893         if(c.hidden){
52894             this.hide();
52895         }
52896     },
52897     /**
52898      * Returns true if this region is currently visible.
52899      * @return {Boolean}
52900      */
52901     isVisible : function(){
52902         return this.visible;
52903     },
52904
52905     /**
52906      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52907      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52908      */
52909     setCollapsedTitle : function(title){
52910         title = title || "&#160;";
52911         if(this.collapsedTitleTextEl){
52912             this.collapsedTitleTextEl.innerHTML = title;
52913         }
52914     },
52915
52916     getBox : function(){
52917         var b;
52918         if(!this.collapsed){
52919             b = this.el.getBox(false, true);
52920         }else{
52921             b = this.collapsedEl.getBox(false, true);
52922         }
52923         return b;
52924     },
52925
52926     getMargins : function(){
52927         return this.collapsed ? this.cmargins : this.margins;
52928     },
52929
52930     highlight : function(){
52931         this.el.addClass("x-layout-panel-dragover");
52932     },
52933
52934     unhighlight : function(){
52935         this.el.removeClass("x-layout-panel-dragover");
52936     },
52937
52938     updateBox : function(box){
52939         this.box = box;
52940         if(!this.collapsed){
52941             this.el.dom.style.left = box.x + "px";
52942             this.el.dom.style.top = box.y + "px";
52943             this.updateBody(box.width, box.height);
52944         }else{
52945             this.collapsedEl.dom.style.left = box.x + "px";
52946             this.collapsedEl.dom.style.top = box.y + "px";
52947             this.collapsedEl.setSize(box.width, box.height);
52948         }
52949         if(this.tabs){
52950             this.tabs.autoSizeTabs();
52951         }
52952     },
52953
52954     updateBody : function(w, h){
52955         if(w !== null){
52956             this.el.setWidth(w);
52957             w -= this.el.getBorderWidth("rl");
52958             if(this.config.adjustments){
52959                 w += this.config.adjustments[0];
52960             }
52961         }
52962         if(h !== null){
52963             this.el.setHeight(h);
52964             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52965             h -= this.el.getBorderWidth("tb");
52966             if(this.config.adjustments){
52967                 h += this.config.adjustments[1];
52968             }
52969             this.bodyEl.setHeight(h);
52970             if(this.tabs){
52971                 h = this.tabs.syncHeight(h);
52972             }
52973         }
52974         if(this.panelSize){
52975             w = w !== null ? w : this.panelSize.width;
52976             h = h !== null ? h : this.panelSize.height;
52977         }
52978         if(this.activePanel){
52979             var el = this.activePanel.getEl();
52980             w = w !== null ? w : el.getWidth();
52981             h = h !== null ? h : el.getHeight();
52982             this.panelSize = {width: w, height: h};
52983             this.activePanel.setSize(w, h);
52984         }
52985         if(Roo.isIE && this.tabs){
52986             this.tabs.el.repaint();
52987         }
52988     },
52989
52990     /**
52991      * Returns the container element for this region.
52992      * @return {Roo.Element}
52993      */
52994     getEl : function(){
52995         return this.el;
52996     },
52997
52998     /**
52999      * Hides this region.
53000      */
53001     hide : function(){
53002         if(!this.collapsed){
53003             this.el.dom.style.left = "-2000px";
53004             this.el.hide();
53005         }else{
53006             this.collapsedEl.dom.style.left = "-2000px";
53007             this.collapsedEl.hide();
53008         }
53009         this.visible = false;
53010         this.fireEvent("visibilitychange", this, false);
53011     },
53012
53013     /**
53014      * Shows this region if it was previously hidden.
53015      */
53016     show : function(){
53017         if(!this.collapsed){
53018             this.el.show();
53019         }else{
53020             this.collapsedEl.show();
53021         }
53022         this.visible = true;
53023         this.fireEvent("visibilitychange", this, true);
53024     },
53025
53026     closeClicked : function(){
53027         if(this.activePanel){
53028             this.remove(this.activePanel);
53029         }
53030     },
53031
53032     collapseClick : function(e){
53033         if(this.isSlid){
53034            e.stopPropagation();
53035            this.slideIn();
53036         }else{
53037            e.stopPropagation();
53038            this.slideOut();
53039         }
53040     },
53041
53042     /**
53043      * Collapses this region.
53044      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53045      */
53046     collapse : function(skipAnim, skipCheck){
53047         if(this.collapsed) {
53048             return;
53049         }
53050         
53051         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53052             
53053             this.collapsed = true;
53054             if(this.split){
53055                 this.split.el.hide();
53056             }
53057             if(this.config.animate && skipAnim !== true){
53058                 this.fireEvent("invalidated", this);
53059                 this.animateCollapse();
53060             }else{
53061                 this.el.setLocation(-20000,-20000);
53062                 this.el.hide();
53063                 this.collapsedEl.show();
53064                 this.fireEvent("collapsed", this);
53065                 this.fireEvent("invalidated", this);
53066             }
53067         }
53068         
53069     },
53070
53071     animateCollapse : function(){
53072         // overridden
53073     },
53074
53075     /**
53076      * Expands this region if it was previously collapsed.
53077      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53078      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53079      */
53080     expand : function(e, skipAnim){
53081         if(e) {
53082             e.stopPropagation();
53083         }
53084         if(!this.collapsed || this.el.hasActiveFx()) {
53085             return;
53086         }
53087         if(this.isSlid){
53088             this.afterSlideIn();
53089             skipAnim = true;
53090         }
53091         this.collapsed = false;
53092         if(this.config.animate && skipAnim !== true){
53093             this.animateExpand();
53094         }else{
53095             this.el.show();
53096             if(this.split){
53097                 this.split.el.show();
53098             }
53099             this.collapsedEl.setLocation(-2000,-2000);
53100             this.collapsedEl.hide();
53101             this.fireEvent("invalidated", this);
53102             this.fireEvent("expanded", this);
53103         }
53104     },
53105
53106     animateExpand : function(){
53107         // overridden
53108     },
53109
53110     initTabs : function()
53111     {
53112         this.bodyEl.setStyle("overflow", "hidden");
53113         var ts = new Roo.TabPanel(
53114                 this.bodyEl.dom,
53115                 {
53116                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53117                     disableTooltips: this.config.disableTabTips,
53118                     toolbar : this.config.toolbar
53119                 }
53120         );
53121         if(this.config.hideTabs){
53122             ts.stripWrap.setDisplayed(false);
53123         }
53124         this.tabs = ts;
53125         ts.resizeTabs = this.config.resizeTabs === true;
53126         ts.minTabWidth = this.config.minTabWidth || 40;
53127         ts.maxTabWidth = this.config.maxTabWidth || 250;
53128         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53129         ts.monitorResize = false;
53130         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53131         ts.bodyEl.addClass('x-layout-tabs-body');
53132         this.panels.each(this.initPanelAsTab, this);
53133     },
53134
53135     initPanelAsTab : function(panel){
53136         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53137                     this.config.closeOnTab && panel.isClosable());
53138         if(panel.tabTip !== undefined){
53139             ti.setTooltip(panel.tabTip);
53140         }
53141         ti.on("activate", function(){
53142               this.setActivePanel(panel);
53143         }, this);
53144         if(this.config.closeOnTab){
53145             ti.on("beforeclose", function(t, e){
53146                 e.cancel = true;
53147                 this.remove(panel);
53148             }, this);
53149         }
53150         return ti;
53151     },
53152
53153     updatePanelTitle : function(panel, title){
53154         if(this.activePanel == panel){
53155             this.updateTitle(title);
53156         }
53157         if(this.tabs){
53158             var ti = this.tabs.getTab(panel.getEl().id);
53159             ti.setText(title);
53160             if(panel.tabTip !== undefined){
53161                 ti.setTooltip(panel.tabTip);
53162             }
53163         }
53164     },
53165
53166     updateTitle : function(title){
53167         if(this.titleTextEl && !this.config.title){
53168             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53169         }
53170     },
53171
53172     setActivePanel : function(panel){
53173         panel = this.getPanel(panel);
53174         if(this.activePanel && this.activePanel != panel){
53175             this.activePanel.setActiveState(false);
53176         }
53177         this.activePanel = panel;
53178         panel.setActiveState(true);
53179         if(this.panelSize){
53180             panel.setSize(this.panelSize.width, this.panelSize.height);
53181         }
53182         if(this.closeBtn){
53183             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53184         }
53185         this.updateTitle(panel.getTitle());
53186         if(this.tabs){
53187             this.fireEvent("invalidated", this);
53188         }
53189         this.fireEvent("panelactivated", this, panel);
53190     },
53191
53192     /**
53193      * Shows the specified panel.
53194      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53195      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53196      */
53197     showPanel : function(panel)
53198     {
53199         panel = this.getPanel(panel);
53200         if(panel){
53201             if(this.tabs){
53202                 var tab = this.tabs.getTab(panel.getEl().id);
53203                 if(tab.isHidden()){
53204                     this.tabs.unhideTab(tab.id);
53205                 }
53206                 tab.activate();
53207             }else{
53208                 this.setActivePanel(panel);
53209             }
53210         }
53211         return panel;
53212     },
53213
53214     /**
53215      * Get the active panel for this region.
53216      * @return {Roo.ContentPanel} The active panel or null
53217      */
53218     getActivePanel : function(){
53219         return this.activePanel;
53220     },
53221
53222     validateVisibility : function(){
53223         if(this.panels.getCount() < 1){
53224             this.updateTitle("&#160;");
53225             this.closeBtn.hide();
53226             this.hide();
53227         }else{
53228             if(!this.isVisible()){
53229                 this.show();
53230             }
53231         }
53232     },
53233
53234     /**
53235      * Adds the passed ContentPanel(s) to this region.
53236      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53237      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53238      */
53239     add : function(panel){
53240         if(arguments.length > 1){
53241             for(var i = 0, len = arguments.length; i < len; i++) {
53242                 this.add(arguments[i]);
53243             }
53244             return null;
53245         }
53246         if(this.hasPanel(panel)){
53247             this.showPanel(panel);
53248             return panel;
53249         }
53250         panel.setRegion(this);
53251         this.panels.add(panel);
53252         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53253             this.bodyEl.dom.appendChild(panel.getEl().dom);
53254             if(panel.background !== true){
53255                 this.setActivePanel(panel);
53256             }
53257             this.fireEvent("paneladded", this, panel);
53258             return panel;
53259         }
53260         if(!this.tabs){
53261             this.initTabs();
53262         }else{
53263             this.initPanelAsTab(panel);
53264         }
53265         if(panel.background !== true){
53266             this.tabs.activate(panel.getEl().id);
53267         }
53268         this.fireEvent("paneladded", this, panel);
53269         return panel;
53270     },
53271
53272     /**
53273      * Hides the tab for the specified panel.
53274      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53275      */
53276     hidePanel : function(panel){
53277         if(this.tabs && (panel = this.getPanel(panel))){
53278             this.tabs.hideTab(panel.getEl().id);
53279         }
53280     },
53281
53282     /**
53283      * Unhides the tab for a previously hidden panel.
53284      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53285      */
53286     unhidePanel : function(panel){
53287         if(this.tabs && (panel = this.getPanel(panel))){
53288             this.tabs.unhideTab(panel.getEl().id);
53289         }
53290     },
53291
53292     clearPanels : function(){
53293         while(this.panels.getCount() > 0){
53294              this.remove(this.panels.first());
53295         }
53296     },
53297
53298     /**
53299      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53300      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53301      * @param {Boolean} preservePanel Overrides the config preservePanel option
53302      * @return {Roo.ContentPanel} The panel that was removed
53303      */
53304     remove : function(panel, preservePanel){
53305         panel = this.getPanel(panel);
53306         if(!panel){
53307             return null;
53308         }
53309         var e = {};
53310         this.fireEvent("beforeremove", this, panel, e);
53311         if(e.cancel === true){
53312             return null;
53313         }
53314         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53315         var panelId = panel.getId();
53316         this.panels.removeKey(panelId);
53317         if(preservePanel){
53318             document.body.appendChild(panel.getEl().dom);
53319         }
53320         if(this.tabs){
53321             this.tabs.removeTab(panel.getEl().id);
53322         }else if (!preservePanel){
53323             this.bodyEl.dom.removeChild(panel.getEl().dom);
53324         }
53325         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53326             var p = this.panels.first();
53327             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53328             tempEl.appendChild(p.getEl().dom);
53329             this.bodyEl.update("");
53330             this.bodyEl.dom.appendChild(p.getEl().dom);
53331             tempEl = null;
53332             this.updateTitle(p.getTitle());
53333             this.tabs = null;
53334             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53335             this.setActivePanel(p);
53336         }
53337         panel.setRegion(null);
53338         if(this.activePanel == panel){
53339             this.activePanel = null;
53340         }
53341         if(this.config.autoDestroy !== false && preservePanel !== true){
53342             try{panel.destroy();}catch(e){}
53343         }
53344         this.fireEvent("panelremoved", this, panel);
53345         return panel;
53346     },
53347
53348     /**
53349      * Returns the TabPanel component used by this region
53350      * @return {Roo.TabPanel}
53351      */
53352     getTabs : function(){
53353         return this.tabs;
53354     },
53355
53356     createTool : function(parentEl, className){
53357         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53358             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53359         btn.addClassOnOver("x-layout-tools-button-over");
53360         return btn;
53361     }
53362 });/*
53363  * Based on:
53364  * Ext JS Library 1.1.1
53365  * Copyright(c) 2006-2007, Ext JS, LLC.
53366  *
53367  * Originally Released Under LGPL - original licence link has changed is not relivant.
53368  *
53369  * Fork - LGPL
53370  * <script type="text/javascript">
53371  */
53372  
53373
53374
53375 /**
53376  * @class Roo.SplitLayoutRegion
53377  * @extends Roo.LayoutRegion
53378  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53379  */
53380 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53381     this.cursor = cursor;
53382     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53383 };
53384
53385 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53386     splitTip : "Drag to resize.",
53387     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53388     useSplitTips : false,
53389
53390     applyConfig : function(config){
53391         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53392         if(config.split){
53393             if(!this.split){
53394                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53395                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53396                 /** The SplitBar for this region 
53397                 * @type Roo.SplitBar */
53398                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53399                 this.split.on("moved", this.onSplitMove, this);
53400                 this.split.useShim = config.useShim === true;
53401                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53402                 if(this.useSplitTips){
53403                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53404                 }
53405                 if(config.collapsible){
53406                     this.split.el.on("dblclick", this.collapse,  this);
53407                 }
53408             }
53409             if(typeof config.minSize != "undefined"){
53410                 this.split.minSize = config.minSize;
53411             }
53412             if(typeof config.maxSize != "undefined"){
53413                 this.split.maxSize = config.maxSize;
53414             }
53415             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53416                 this.hideSplitter();
53417             }
53418         }
53419     },
53420
53421     getHMaxSize : function(){
53422          var cmax = this.config.maxSize || 10000;
53423          var center = this.mgr.getRegion("center");
53424          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53425     },
53426
53427     getVMaxSize : function(){
53428          var cmax = this.config.maxSize || 10000;
53429          var center = this.mgr.getRegion("center");
53430          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53431     },
53432
53433     onSplitMove : function(split, newSize){
53434         this.fireEvent("resized", this, newSize);
53435     },
53436     
53437     /** 
53438      * Returns the {@link Roo.SplitBar} for this region.
53439      * @return {Roo.SplitBar}
53440      */
53441     getSplitBar : function(){
53442         return this.split;
53443     },
53444     
53445     hide : function(){
53446         this.hideSplitter();
53447         Roo.SplitLayoutRegion.superclass.hide.call(this);
53448     },
53449
53450     hideSplitter : function(){
53451         if(this.split){
53452             this.split.el.setLocation(-2000,-2000);
53453             this.split.el.hide();
53454         }
53455     },
53456
53457     show : function(){
53458         if(this.split){
53459             this.split.el.show();
53460         }
53461         Roo.SplitLayoutRegion.superclass.show.call(this);
53462     },
53463     
53464     beforeSlide: function(){
53465         if(Roo.isGecko){// firefox overflow auto bug workaround
53466             this.bodyEl.clip();
53467             if(this.tabs) {
53468                 this.tabs.bodyEl.clip();
53469             }
53470             if(this.activePanel){
53471                 this.activePanel.getEl().clip();
53472                 
53473                 if(this.activePanel.beforeSlide){
53474                     this.activePanel.beforeSlide();
53475                 }
53476             }
53477         }
53478     },
53479     
53480     afterSlide : function(){
53481         if(Roo.isGecko){// firefox overflow auto bug workaround
53482             this.bodyEl.unclip();
53483             if(this.tabs) {
53484                 this.tabs.bodyEl.unclip();
53485             }
53486             if(this.activePanel){
53487                 this.activePanel.getEl().unclip();
53488                 if(this.activePanel.afterSlide){
53489                     this.activePanel.afterSlide();
53490                 }
53491             }
53492         }
53493     },
53494
53495     initAutoHide : function(){
53496         if(this.autoHide !== false){
53497             if(!this.autoHideHd){
53498                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53499                 this.autoHideHd = {
53500                     "mouseout": function(e){
53501                         if(!e.within(this.el, true)){
53502                             st.delay(500);
53503                         }
53504                     },
53505                     "mouseover" : function(e){
53506                         st.cancel();
53507                     },
53508                     scope : this
53509                 };
53510             }
53511             this.el.on(this.autoHideHd);
53512         }
53513     },
53514
53515     clearAutoHide : function(){
53516         if(this.autoHide !== false){
53517             this.el.un("mouseout", this.autoHideHd.mouseout);
53518             this.el.un("mouseover", this.autoHideHd.mouseover);
53519         }
53520     },
53521
53522     clearMonitor : function(){
53523         Roo.get(document).un("click", this.slideInIf, this);
53524     },
53525
53526     // these names are backwards but not changed for compat
53527     slideOut : function(){
53528         if(this.isSlid || this.el.hasActiveFx()){
53529             return;
53530         }
53531         this.isSlid = true;
53532         if(this.collapseBtn){
53533             this.collapseBtn.hide();
53534         }
53535         this.closeBtnState = this.closeBtn.getStyle('display');
53536         this.closeBtn.hide();
53537         if(this.stickBtn){
53538             this.stickBtn.show();
53539         }
53540         this.el.show();
53541         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53542         this.beforeSlide();
53543         this.el.setStyle("z-index", 10001);
53544         this.el.slideIn(this.getSlideAnchor(), {
53545             callback: function(){
53546                 this.afterSlide();
53547                 this.initAutoHide();
53548                 Roo.get(document).on("click", this.slideInIf, this);
53549                 this.fireEvent("slideshow", this);
53550             },
53551             scope: this,
53552             block: true
53553         });
53554     },
53555
53556     afterSlideIn : function(){
53557         this.clearAutoHide();
53558         this.isSlid = false;
53559         this.clearMonitor();
53560         this.el.setStyle("z-index", "");
53561         if(this.collapseBtn){
53562             this.collapseBtn.show();
53563         }
53564         this.closeBtn.setStyle('display', this.closeBtnState);
53565         if(this.stickBtn){
53566             this.stickBtn.hide();
53567         }
53568         this.fireEvent("slidehide", this);
53569     },
53570
53571     slideIn : function(cb){
53572         if(!this.isSlid || this.el.hasActiveFx()){
53573             Roo.callback(cb);
53574             return;
53575         }
53576         this.isSlid = false;
53577         this.beforeSlide();
53578         this.el.slideOut(this.getSlideAnchor(), {
53579             callback: function(){
53580                 this.el.setLeftTop(-10000, -10000);
53581                 this.afterSlide();
53582                 this.afterSlideIn();
53583                 Roo.callback(cb);
53584             },
53585             scope: this,
53586             block: true
53587         });
53588     },
53589     
53590     slideInIf : function(e){
53591         if(!e.within(this.el)){
53592             this.slideIn();
53593         }
53594     },
53595
53596     animateCollapse : function(){
53597         this.beforeSlide();
53598         this.el.setStyle("z-index", 20000);
53599         var anchor = this.getSlideAnchor();
53600         this.el.slideOut(anchor, {
53601             callback : function(){
53602                 this.el.setStyle("z-index", "");
53603                 this.collapsedEl.slideIn(anchor, {duration:.3});
53604                 this.afterSlide();
53605                 this.el.setLocation(-10000,-10000);
53606                 this.el.hide();
53607                 this.fireEvent("collapsed", this);
53608             },
53609             scope: this,
53610             block: true
53611         });
53612     },
53613
53614     animateExpand : function(){
53615         this.beforeSlide();
53616         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53617         this.el.setStyle("z-index", 20000);
53618         this.collapsedEl.hide({
53619             duration:.1
53620         });
53621         this.el.slideIn(this.getSlideAnchor(), {
53622             callback : function(){
53623                 this.el.setStyle("z-index", "");
53624                 this.afterSlide();
53625                 if(this.split){
53626                     this.split.el.show();
53627                 }
53628                 this.fireEvent("invalidated", this);
53629                 this.fireEvent("expanded", this);
53630             },
53631             scope: this,
53632             block: true
53633         });
53634     },
53635
53636     anchors : {
53637         "west" : "left",
53638         "east" : "right",
53639         "north" : "top",
53640         "south" : "bottom"
53641     },
53642
53643     sanchors : {
53644         "west" : "l",
53645         "east" : "r",
53646         "north" : "t",
53647         "south" : "b"
53648     },
53649
53650     canchors : {
53651         "west" : "tl-tr",
53652         "east" : "tr-tl",
53653         "north" : "tl-bl",
53654         "south" : "bl-tl"
53655     },
53656
53657     getAnchor : function(){
53658         return this.anchors[this.position];
53659     },
53660
53661     getCollapseAnchor : function(){
53662         return this.canchors[this.position];
53663     },
53664
53665     getSlideAnchor : function(){
53666         return this.sanchors[this.position];
53667     },
53668
53669     getAlignAdj : function(){
53670         var cm = this.cmargins;
53671         switch(this.position){
53672             case "west":
53673                 return [0, 0];
53674             break;
53675             case "east":
53676                 return [0, 0];
53677             break;
53678             case "north":
53679                 return [0, 0];
53680             break;
53681             case "south":
53682                 return [0, 0];
53683             break;
53684         }
53685     },
53686
53687     getExpandAdj : function(){
53688         var c = this.collapsedEl, cm = this.cmargins;
53689         switch(this.position){
53690             case "west":
53691                 return [-(cm.right+c.getWidth()+cm.left), 0];
53692             break;
53693             case "east":
53694                 return [cm.right+c.getWidth()+cm.left, 0];
53695             break;
53696             case "north":
53697                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53698             break;
53699             case "south":
53700                 return [0, cm.top+cm.bottom+c.getHeight()];
53701             break;
53702         }
53703     }
53704 });/*
53705  * Based on:
53706  * Ext JS Library 1.1.1
53707  * Copyright(c) 2006-2007, Ext JS, LLC.
53708  *
53709  * Originally Released Under LGPL - original licence link has changed is not relivant.
53710  *
53711  * Fork - LGPL
53712  * <script type="text/javascript">
53713  */
53714 /*
53715  * These classes are private internal classes
53716  */
53717 Roo.CenterLayoutRegion = function(mgr, config){
53718     Roo.LayoutRegion.call(this, mgr, config, "center");
53719     this.visible = true;
53720     this.minWidth = config.minWidth || 20;
53721     this.minHeight = config.minHeight || 20;
53722 };
53723
53724 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53725     hide : function(){
53726         // center panel can't be hidden
53727     },
53728     
53729     show : function(){
53730         // center panel can't be hidden
53731     },
53732     
53733     getMinWidth: function(){
53734         return this.minWidth;
53735     },
53736     
53737     getMinHeight: function(){
53738         return this.minHeight;
53739     }
53740 });
53741
53742
53743 Roo.NorthLayoutRegion = function(mgr, config){
53744     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53745     if(this.split){
53746         this.split.placement = Roo.SplitBar.TOP;
53747         this.split.orientation = Roo.SplitBar.VERTICAL;
53748         this.split.el.addClass("x-layout-split-v");
53749     }
53750     var size = config.initialSize || config.height;
53751     if(typeof size != "undefined"){
53752         this.el.setHeight(size);
53753     }
53754 };
53755 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53756     orientation: Roo.SplitBar.VERTICAL,
53757     getBox : function(){
53758         if(this.collapsed){
53759             return this.collapsedEl.getBox();
53760         }
53761         var box = this.el.getBox();
53762         if(this.split){
53763             box.height += this.split.el.getHeight();
53764         }
53765         return box;
53766     },
53767     
53768     updateBox : function(box){
53769         if(this.split && !this.collapsed){
53770             box.height -= this.split.el.getHeight();
53771             this.split.el.setLeft(box.x);
53772             this.split.el.setTop(box.y+box.height);
53773             this.split.el.setWidth(box.width);
53774         }
53775         if(this.collapsed){
53776             this.updateBody(box.width, null);
53777         }
53778         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53779     }
53780 });
53781
53782 Roo.SouthLayoutRegion = function(mgr, config){
53783     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53784     if(this.split){
53785         this.split.placement = Roo.SplitBar.BOTTOM;
53786         this.split.orientation = Roo.SplitBar.VERTICAL;
53787         this.split.el.addClass("x-layout-split-v");
53788     }
53789     var size = config.initialSize || config.height;
53790     if(typeof size != "undefined"){
53791         this.el.setHeight(size);
53792     }
53793 };
53794 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53795     orientation: Roo.SplitBar.VERTICAL,
53796     getBox : function(){
53797         if(this.collapsed){
53798             return this.collapsedEl.getBox();
53799         }
53800         var box = this.el.getBox();
53801         if(this.split){
53802             var sh = this.split.el.getHeight();
53803             box.height += sh;
53804             box.y -= sh;
53805         }
53806         return box;
53807     },
53808     
53809     updateBox : function(box){
53810         if(this.split && !this.collapsed){
53811             var sh = this.split.el.getHeight();
53812             box.height -= sh;
53813             box.y += sh;
53814             this.split.el.setLeft(box.x);
53815             this.split.el.setTop(box.y-sh);
53816             this.split.el.setWidth(box.width);
53817         }
53818         if(this.collapsed){
53819             this.updateBody(box.width, null);
53820         }
53821         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53822     }
53823 });
53824
53825 Roo.EastLayoutRegion = function(mgr, config){
53826     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53827     if(this.split){
53828         this.split.placement = Roo.SplitBar.RIGHT;
53829         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53830         this.split.el.addClass("x-layout-split-h");
53831     }
53832     var size = config.initialSize || config.width;
53833     if(typeof size != "undefined"){
53834         this.el.setWidth(size);
53835     }
53836 };
53837 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53838     orientation: Roo.SplitBar.HORIZONTAL,
53839     getBox : function(){
53840         if(this.collapsed){
53841             return this.collapsedEl.getBox();
53842         }
53843         var box = this.el.getBox();
53844         if(this.split){
53845             var sw = this.split.el.getWidth();
53846             box.width += sw;
53847             box.x -= sw;
53848         }
53849         return box;
53850     },
53851
53852     updateBox : function(box){
53853         if(this.split && !this.collapsed){
53854             var sw = this.split.el.getWidth();
53855             box.width -= sw;
53856             this.split.el.setLeft(box.x);
53857             this.split.el.setTop(box.y);
53858             this.split.el.setHeight(box.height);
53859             box.x += sw;
53860         }
53861         if(this.collapsed){
53862             this.updateBody(null, box.height);
53863         }
53864         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53865     }
53866 });
53867
53868 Roo.WestLayoutRegion = function(mgr, config){
53869     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53870     if(this.split){
53871         this.split.placement = Roo.SplitBar.LEFT;
53872         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53873         this.split.el.addClass("x-layout-split-h");
53874     }
53875     var size = config.initialSize || config.width;
53876     if(typeof size != "undefined"){
53877         this.el.setWidth(size);
53878     }
53879 };
53880 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53881     orientation: Roo.SplitBar.HORIZONTAL,
53882     getBox : function(){
53883         if(this.collapsed){
53884             return this.collapsedEl.getBox();
53885         }
53886         var box = this.el.getBox();
53887         if(this.split){
53888             box.width += this.split.el.getWidth();
53889         }
53890         return box;
53891     },
53892     
53893     updateBox : function(box){
53894         if(this.split && !this.collapsed){
53895             var sw = this.split.el.getWidth();
53896             box.width -= sw;
53897             this.split.el.setLeft(box.x+box.width);
53898             this.split.el.setTop(box.y);
53899             this.split.el.setHeight(box.height);
53900         }
53901         if(this.collapsed){
53902             this.updateBody(null, box.height);
53903         }
53904         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53905     }
53906 });
53907 /*
53908  * Based on:
53909  * Ext JS Library 1.1.1
53910  * Copyright(c) 2006-2007, Ext JS, LLC.
53911  *
53912  * Originally Released Under LGPL - original licence link has changed is not relivant.
53913  *
53914  * Fork - LGPL
53915  * <script type="text/javascript">
53916  */
53917  
53918  
53919 /*
53920  * Private internal class for reading and applying state
53921  */
53922 Roo.LayoutStateManager = function(layout){
53923      // default empty state
53924      this.state = {
53925         north: {},
53926         south: {},
53927         east: {},
53928         west: {}       
53929     };
53930 };
53931
53932 Roo.LayoutStateManager.prototype = {
53933     init : function(layout, provider){
53934         this.provider = provider;
53935         var state = provider.get(layout.id+"-layout-state");
53936         if(state){
53937             var wasUpdating = layout.isUpdating();
53938             if(!wasUpdating){
53939                 layout.beginUpdate();
53940             }
53941             for(var key in state){
53942                 if(typeof state[key] != "function"){
53943                     var rstate = state[key];
53944                     var r = layout.getRegion(key);
53945                     if(r && rstate){
53946                         if(rstate.size){
53947                             r.resizeTo(rstate.size);
53948                         }
53949                         if(rstate.collapsed == true){
53950                             r.collapse(true);
53951                         }else{
53952                             r.expand(null, true);
53953                         }
53954                     }
53955                 }
53956             }
53957             if(!wasUpdating){
53958                 layout.endUpdate();
53959             }
53960             this.state = state; 
53961         }
53962         this.layout = layout;
53963         layout.on("regionresized", this.onRegionResized, this);
53964         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53965         layout.on("regionexpanded", this.onRegionExpanded, this);
53966     },
53967     
53968     storeState : function(){
53969         this.provider.set(this.layout.id+"-layout-state", this.state);
53970     },
53971     
53972     onRegionResized : function(region, newSize){
53973         this.state[region.getPosition()].size = newSize;
53974         this.storeState();
53975     },
53976     
53977     onRegionCollapsed : function(region){
53978         this.state[region.getPosition()].collapsed = true;
53979         this.storeState();
53980     },
53981     
53982     onRegionExpanded : function(region){
53983         this.state[region.getPosition()].collapsed = false;
53984         this.storeState();
53985     }
53986 };/*
53987  * Based on:
53988  * Ext JS Library 1.1.1
53989  * Copyright(c) 2006-2007, Ext JS, LLC.
53990  *
53991  * Originally Released Under LGPL - original licence link has changed is not relivant.
53992  *
53993  * Fork - LGPL
53994  * <script type="text/javascript">
53995  */
53996 /**
53997  * @class Roo.ContentPanel
53998  * @extends Roo.util.Observable
53999  * A basic ContentPanel element.
54000  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
54001  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
54002  * @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
54003  * @cfg {Boolean}   closable      True if the panel can be closed/removed
54004  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
54005  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
54006  * @cfg {Toolbar}   toolbar       A toolbar for this panel
54007  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
54008  * @cfg {String} title          The title for this panel
54009  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
54010  * @cfg {String} url            Calls {@link #setUrl} with this value
54011  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
54012  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
54013  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
54014  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
54015
54016  * @constructor
54017  * Create a new ContentPanel.
54018  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54019  * @param {String/Object} config A string to set only the title or a config object
54020  * @param {String} content (optional) Set the HTML content for this panel
54021  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54022  */
54023 Roo.ContentPanel = function(el, config, content){
54024     
54025      
54026     /*
54027     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54028         config = el;
54029         el = Roo.id();
54030     }
54031     if (config && config.parentLayout) { 
54032         el = config.parentLayout.el.createChild(); 
54033     }
54034     */
54035     if(el.autoCreate){ // xtype is available if this is called from factory
54036         config = el;
54037         el = Roo.id();
54038     }
54039     this.el = Roo.get(el);
54040     if(!this.el && config && config.autoCreate){
54041         if(typeof config.autoCreate == "object"){
54042             if(!config.autoCreate.id){
54043                 config.autoCreate.id = config.id||el;
54044             }
54045             this.el = Roo.DomHelper.append(document.body,
54046                         config.autoCreate, true);
54047         }else{
54048             this.el = Roo.DomHelper.append(document.body,
54049                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54050         }
54051     }
54052     this.closable = false;
54053     this.loaded = false;
54054     this.active = false;
54055     if(typeof config == "string"){
54056         this.title = config;
54057     }else{
54058         Roo.apply(this, config);
54059     }
54060     
54061     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54062         this.wrapEl = this.el.wrap();
54063         this.toolbar.container = this.el.insertSibling(false, 'before');
54064         this.toolbar = new Roo.Toolbar(this.toolbar);
54065     }
54066     
54067     // xtype created footer. - not sure if will work as we normally have to render first..
54068     if (this.footer && !this.footer.el && this.footer.xtype) {
54069         if (!this.wrapEl) {
54070             this.wrapEl = this.el.wrap();
54071         }
54072     
54073         this.footer.container = this.wrapEl.createChild();
54074          
54075         this.footer = Roo.factory(this.footer, Roo);
54076         
54077     }
54078     
54079     if(this.resizeEl){
54080         this.resizeEl = Roo.get(this.resizeEl, true);
54081     }else{
54082         this.resizeEl = this.el;
54083     }
54084     // handle view.xtype
54085     
54086  
54087     
54088     
54089     this.addEvents({
54090         /**
54091          * @event activate
54092          * Fires when this panel is activated. 
54093          * @param {Roo.ContentPanel} this
54094          */
54095         "activate" : true,
54096         /**
54097          * @event deactivate
54098          * Fires when this panel is activated. 
54099          * @param {Roo.ContentPanel} this
54100          */
54101         "deactivate" : true,
54102
54103         /**
54104          * @event resize
54105          * Fires when this panel is resized if fitToFrame is true.
54106          * @param {Roo.ContentPanel} this
54107          * @param {Number} width The width after any component adjustments
54108          * @param {Number} height The height after any component adjustments
54109          */
54110         "resize" : true,
54111         
54112          /**
54113          * @event render
54114          * Fires when this tab is created
54115          * @param {Roo.ContentPanel} this
54116          */
54117         "render" : true
54118          
54119         
54120     });
54121     
54122
54123     
54124     
54125     if(this.autoScroll){
54126         this.resizeEl.setStyle("overflow", "auto");
54127     } else {
54128         // fix randome scrolling
54129         this.el.on('scroll', function() {
54130             Roo.log('fix random scolling');
54131             this.scrollTo('top',0); 
54132         });
54133     }
54134     content = content || this.content;
54135     if(content){
54136         this.setContent(content);
54137     }
54138     if(config && config.url){
54139         this.setUrl(this.url, this.params, this.loadOnce);
54140     }
54141     
54142     
54143     
54144     Roo.ContentPanel.superclass.constructor.call(this);
54145     
54146     if (this.view && typeof(this.view.xtype) != 'undefined') {
54147         this.view.el = this.el.appendChild(document.createElement("div"));
54148         this.view = Roo.factory(this.view); 
54149         this.view.render  &&  this.view.render(false, '');  
54150     }
54151     
54152     
54153     this.fireEvent('render', this);
54154 };
54155
54156 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54157     tabTip:'',
54158     setRegion : function(region){
54159         this.region = region;
54160         if(region){
54161            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54162         }else{
54163            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54164         } 
54165     },
54166     
54167     /**
54168      * Returns the toolbar for this Panel if one was configured. 
54169      * @return {Roo.Toolbar} 
54170      */
54171     getToolbar : function(){
54172         return this.toolbar;
54173     },
54174     
54175     setActiveState : function(active){
54176         this.active = active;
54177         if(!active){
54178             this.fireEvent("deactivate", this);
54179         }else{
54180             this.fireEvent("activate", this);
54181         }
54182     },
54183     /**
54184      * Updates this panel's element
54185      * @param {String} content The new content
54186      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54187     */
54188     setContent : function(content, loadScripts){
54189         this.el.update(content, loadScripts);
54190     },
54191
54192     ignoreResize : function(w, h){
54193         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54194             return true;
54195         }else{
54196             this.lastSize = {width: w, height: h};
54197             return false;
54198         }
54199     },
54200     /**
54201      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54202      * @return {Roo.UpdateManager} The UpdateManager
54203      */
54204     getUpdateManager : function(){
54205         return this.el.getUpdateManager();
54206     },
54207      /**
54208      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54209      * @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:
54210 <pre><code>
54211 panel.load({
54212     url: "your-url.php",
54213     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54214     callback: yourFunction,
54215     scope: yourObject, //(optional scope)
54216     discardUrl: false,
54217     nocache: false,
54218     text: "Loading...",
54219     timeout: 30,
54220     scripts: false
54221 });
54222 </code></pre>
54223      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54224      * 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.
54225      * @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}
54226      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54227      * @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.
54228      * @return {Roo.ContentPanel} this
54229      */
54230     load : function(){
54231         var um = this.el.getUpdateManager();
54232         um.update.apply(um, arguments);
54233         return this;
54234     },
54235
54236
54237     /**
54238      * 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.
54239      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54240      * @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)
54241      * @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)
54242      * @return {Roo.UpdateManager} The UpdateManager
54243      */
54244     setUrl : function(url, params, loadOnce){
54245         if(this.refreshDelegate){
54246             this.removeListener("activate", this.refreshDelegate);
54247         }
54248         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54249         this.on("activate", this.refreshDelegate);
54250         return this.el.getUpdateManager();
54251     },
54252     
54253     _handleRefresh : function(url, params, loadOnce){
54254         if(!loadOnce || !this.loaded){
54255             var updater = this.el.getUpdateManager();
54256             updater.update(url, params, this._setLoaded.createDelegate(this));
54257         }
54258     },
54259     
54260     _setLoaded : function(){
54261         this.loaded = true;
54262     }, 
54263     
54264     /**
54265      * Returns this panel's id
54266      * @return {String} 
54267      */
54268     getId : function(){
54269         return this.el.id;
54270     },
54271     
54272     /** 
54273      * Returns this panel's element - used by regiosn to add.
54274      * @return {Roo.Element} 
54275      */
54276     getEl : function(){
54277         return this.wrapEl || this.el;
54278     },
54279     
54280     adjustForComponents : function(width, height)
54281     {
54282         //Roo.log('adjustForComponents ');
54283         if(this.resizeEl != this.el){
54284             width -= this.el.getFrameWidth('lr');
54285             height -= this.el.getFrameWidth('tb');
54286         }
54287         if(this.toolbar){
54288             var te = this.toolbar.getEl();
54289             height -= te.getHeight();
54290             te.setWidth(width);
54291         }
54292         if(this.footer){
54293             var te = this.footer.getEl();
54294             //Roo.log("footer:" + te.getHeight());
54295             
54296             height -= te.getHeight();
54297             te.setWidth(width);
54298         }
54299         
54300         
54301         if(this.adjustments){
54302             width += this.adjustments[0];
54303             height += this.adjustments[1];
54304         }
54305         return {"width": width, "height": height};
54306     },
54307     
54308     setSize : function(width, height){
54309         if(this.fitToFrame && !this.ignoreResize(width, height)){
54310             if(this.fitContainer && this.resizeEl != this.el){
54311                 this.el.setSize(width, height);
54312             }
54313             var size = this.adjustForComponents(width, height);
54314             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54315             this.fireEvent('resize', this, size.width, size.height);
54316         }
54317     },
54318     
54319     /**
54320      * Returns this panel's title
54321      * @return {String} 
54322      */
54323     getTitle : function(){
54324         return this.title;
54325     },
54326     
54327     /**
54328      * Set this panel's title
54329      * @param {String} title
54330      */
54331     setTitle : function(title){
54332         this.title = title;
54333         if(this.region){
54334             this.region.updatePanelTitle(this, title);
54335         }
54336     },
54337     
54338     /**
54339      * Returns true is this panel was configured to be closable
54340      * @return {Boolean} 
54341      */
54342     isClosable : function(){
54343         return this.closable;
54344     },
54345     
54346     beforeSlide : function(){
54347         this.el.clip();
54348         this.resizeEl.clip();
54349     },
54350     
54351     afterSlide : function(){
54352         this.el.unclip();
54353         this.resizeEl.unclip();
54354     },
54355     
54356     /**
54357      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54358      *   Will fail silently if the {@link #setUrl} method has not been called.
54359      *   This does not activate the panel, just updates its content.
54360      */
54361     refresh : function(){
54362         if(this.refreshDelegate){
54363            this.loaded = false;
54364            this.refreshDelegate();
54365         }
54366     },
54367     
54368     /**
54369      * Destroys this panel
54370      */
54371     destroy : function(){
54372         this.el.removeAllListeners();
54373         var tempEl = document.createElement("span");
54374         tempEl.appendChild(this.el.dom);
54375         tempEl.innerHTML = "";
54376         this.el.remove();
54377         this.el = null;
54378     },
54379     
54380     /**
54381      * form - if the content panel contains a form - this is a reference to it.
54382      * @type {Roo.form.Form}
54383      */
54384     form : false,
54385     /**
54386      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54387      *    This contains a reference to it.
54388      * @type {Roo.View}
54389      */
54390     view : false,
54391     
54392       /**
54393      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54394      * <pre><code>
54395
54396 layout.addxtype({
54397        xtype : 'Form',
54398        items: [ .... ]
54399    }
54400 );
54401
54402 </code></pre>
54403      * @param {Object} cfg Xtype definition of item to add.
54404      */
54405     
54406     addxtype : function(cfg) {
54407         // add form..
54408         if (cfg.xtype.match(/^Form$/)) {
54409             
54410             var el;
54411             //if (this.footer) {
54412             //    el = this.footer.container.insertSibling(false, 'before');
54413             //} else {
54414                 el = this.el.createChild();
54415             //}
54416
54417             this.form = new  Roo.form.Form(cfg);
54418             
54419             
54420             if ( this.form.allItems.length) {
54421                 this.form.render(el.dom);
54422             }
54423             return this.form;
54424         }
54425         // should only have one of theses..
54426         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54427             // views.. should not be just added - used named prop 'view''
54428             
54429             cfg.el = this.el.appendChild(document.createElement("div"));
54430             // factory?
54431             
54432             var ret = new Roo.factory(cfg);
54433              
54434              ret.render && ret.render(false, ''); // render blank..
54435             this.view = ret;
54436             return ret;
54437         }
54438         return false;
54439     }
54440 });
54441
54442 /**
54443  * @class Roo.GridPanel
54444  * @extends Roo.ContentPanel
54445  * @constructor
54446  * Create a new GridPanel.
54447  * @param {Roo.grid.Grid} grid The grid for this panel
54448  * @param {String/Object} config A string to set only the panel's title, or a config object
54449  */
54450 Roo.GridPanel = function(grid, config){
54451     
54452   
54453     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54454         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54455         
54456     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54457     
54458     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54459     
54460     if(this.toolbar){
54461         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54462     }
54463     // xtype created footer. - not sure if will work as we normally have to render first..
54464     if (this.footer && !this.footer.el && this.footer.xtype) {
54465         
54466         this.footer.container = this.grid.getView().getFooterPanel(true);
54467         this.footer.dataSource = this.grid.dataSource;
54468         this.footer = Roo.factory(this.footer, Roo);
54469         
54470     }
54471     
54472     grid.monitorWindowResize = false; // turn off autosizing
54473     grid.autoHeight = false;
54474     grid.autoWidth = false;
54475     this.grid = grid;
54476     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54477 };
54478
54479 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54480     getId : function(){
54481         return this.grid.id;
54482     },
54483     
54484     /**
54485      * Returns the grid for this panel
54486      * @return {Roo.grid.Grid} 
54487      */
54488     getGrid : function(){
54489         return this.grid;    
54490     },
54491     
54492     setSize : function(width, height){
54493         if(!this.ignoreResize(width, height)){
54494             var grid = this.grid;
54495             var size = this.adjustForComponents(width, height);
54496             grid.getGridEl().setSize(size.width, size.height);
54497             grid.autoSize();
54498         }
54499     },
54500     
54501     beforeSlide : function(){
54502         this.grid.getView().scroller.clip();
54503     },
54504     
54505     afterSlide : function(){
54506         this.grid.getView().scroller.unclip();
54507     },
54508     
54509     destroy : function(){
54510         this.grid.destroy();
54511         delete this.grid;
54512         Roo.GridPanel.superclass.destroy.call(this); 
54513     }
54514 });
54515
54516
54517 /**
54518  * @class Roo.NestedLayoutPanel
54519  * @extends Roo.ContentPanel
54520  * @constructor
54521  * Create a new NestedLayoutPanel.
54522  * 
54523  * 
54524  * @param {Roo.BorderLayout} layout The layout for this panel
54525  * @param {String/Object} config A string to set only the title or a config object
54526  */
54527 Roo.NestedLayoutPanel = function(layout, config)
54528 {
54529     // construct with only one argument..
54530     /* FIXME - implement nicer consturctors
54531     if (layout.layout) {
54532         config = layout;
54533         layout = config.layout;
54534         delete config.layout;
54535     }
54536     if (layout.xtype && !layout.getEl) {
54537         // then layout needs constructing..
54538         layout = Roo.factory(layout, Roo);
54539     }
54540     */
54541     
54542     
54543     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54544     
54545     layout.monitorWindowResize = false; // turn off autosizing
54546     this.layout = layout;
54547     this.layout.getEl().addClass("x-layout-nested-layout");
54548     
54549     
54550     
54551     
54552 };
54553
54554 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54555
54556     setSize : function(width, height){
54557         if(!this.ignoreResize(width, height)){
54558             var size = this.adjustForComponents(width, height);
54559             var el = this.layout.getEl();
54560             el.setSize(size.width, size.height);
54561             var touch = el.dom.offsetWidth;
54562             this.layout.layout();
54563             // ie requires a double layout on the first pass
54564             if(Roo.isIE && !this.initialized){
54565                 this.initialized = true;
54566                 this.layout.layout();
54567             }
54568         }
54569     },
54570     
54571     // activate all subpanels if not currently active..
54572     
54573     setActiveState : function(active){
54574         this.active = active;
54575         if(!active){
54576             this.fireEvent("deactivate", this);
54577             return;
54578         }
54579         
54580         this.fireEvent("activate", this);
54581         // not sure if this should happen before or after..
54582         if (!this.layout) {
54583             return; // should not happen..
54584         }
54585         var reg = false;
54586         for (var r in this.layout.regions) {
54587             reg = this.layout.getRegion(r);
54588             if (reg.getActivePanel()) {
54589                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54590                 reg.setActivePanel(reg.getActivePanel());
54591                 continue;
54592             }
54593             if (!reg.panels.length) {
54594                 continue;
54595             }
54596             reg.showPanel(reg.getPanel(0));
54597         }
54598         
54599         
54600         
54601         
54602     },
54603     
54604     /**
54605      * Returns the nested BorderLayout for this panel
54606      * @return {Roo.BorderLayout} 
54607      */
54608     getLayout : function(){
54609         return this.layout;
54610     },
54611     
54612      /**
54613      * Adds a xtype elements to the layout of the nested panel
54614      * <pre><code>
54615
54616 panel.addxtype({
54617        xtype : 'ContentPanel',
54618        region: 'west',
54619        items: [ .... ]
54620    }
54621 );
54622
54623 panel.addxtype({
54624         xtype : 'NestedLayoutPanel',
54625         region: 'west',
54626         layout: {
54627            center: { },
54628            west: { }   
54629         },
54630         items : [ ... list of content panels or nested layout panels.. ]
54631    }
54632 );
54633 </code></pre>
54634      * @param {Object} cfg Xtype definition of item to add.
54635      */
54636     addxtype : function(cfg) {
54637         return this.layout.addxtype(cfg);
54638     
54639     }
54640 });
54641
54642 Roo.ScrollPanel = function(el, config, content){
54643     config = config || {};
54644     config.fitToFrame = true;
54645     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54646     
54647     this.el.dom.style.overflow = "hidden";
54648     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54649     this.el.removeClass("x-layout-inactive-content");
54650     this.el.on("mousewheel", this.onWheel, this);
54651
54652     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54653     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54654     up.unselectable(); down.unselectable();
54655     up.on("click", this.scrollUp, this);
54656     down.on("click", this.scrollDown, this);
54657     up.addClassOnOver("x-scroller-btn-over");
54658     down.addClassOnOver("x-scroller-btn-over");
54659     up.addClassOnClick("x-scroller-btn-click");
54660     down.addClassOnClick("x-scroller-btn-click");
54661     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54662
54663     this.resizeEl = this.el;
54664     this.el = wrap; this.up = up; this.down = down;
54665 };
54666
54667 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54668     increment : 100,
54669     wheelIncrement : 5,
54670     scrollUp : function(){
54671         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54672     },
54673
54674     scrollDown : function(){
54675         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54676     },
54677
54678     afterScroll : function(){
54679         var el = this.resizeEl;
54680         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54681         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54682         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54683     },
54684
54685     setSize : function(){
54686         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54687         this.afterScroll();
54688     },
54689
54690     onWheel : function(e){
54691         var d = e.getWheelDelta();
54692         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54693         this.afterScroll();
54694         e.stopEvent();
54695     },
54696
54697     setContent : function(content, loadScripts){
54698         this.resizeEl.update(content, loadScripts);
54699     }
54700
54701 });
54702
54703
54704
54705
54706
54707
54708
54709
54710
54711 /**
54712  * @class Roo.TreePanel
54713  * @extends Roo.ContentPanel
54714  * @constructor
54715  * Create a new TreePanel. - defaults to fit/scoll contents.
54716  * @param {String/Object} config A string to set only the panel's title, or a config object
54717  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54718  */
54719 Roo.TreePanel = function(config){
54720     var el = config.el;
54721     var tree = config.tree;
54722     delete config.tree; 
54723     delete config.el; // hopefull!
54724     
54725     // wrapper for IE7 strict & safari scroll issue
54726     
54727     var treeEl = el.createChild();
54728     config.resizeEl = treeEl;
54729     
54730     
54731     
54732     Roo.TreePanel.superclass.constructor.call(this, el, config);
54733  
54734  
54735     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54736     //console.log(tree);
54737     this.on('activate', function()
54738     {
54739         if (this.tree.rendered) {
54740             return;
54741         }
54742         //console.log('render tree');
54743         this.tree.render();
54744     });
54745     // this should not be needed.. - it's actually the 'el' that resizes?
54746     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54747     
54748     //this.on('resize',  function (cp, w, h) {
54749     //        this.tree.innerCt.setWidth(w);
54750     //        this.tree.innerCt.setHeight(h);
54751     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54752     //});
54753
54754         
54755     
54756 };
54757
54758 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54759     fitToFrame : true,
54760     autoScroll : true
54761 });
54762
54763
54764
54765
54766
54767
54768
54769
54770
54771
54772
54773 /*
54774  * Based on:
54775  * Ext JS Library 1.1.1
54776  * Copyright(c) 2006-2007, Ext JS, LLC.
54777  *
54778  * Originally Released Under LGPL - original licence link has changed is not relivant.
54779  *
54780  * Fork - LGPL
54781  * <script type="text/javascript">
54782  */
54783  
54784
54785 /**
54786  * @class Roo.ReaderLayout
54787  * @extends Roo.BorderLayout
54788  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54789  * center region containing two nested regions (a top one for a list view and one for item preview below),
54790  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54791  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54792  * expedites the setup of the overall layout and regions for this common application style.
54793  * Example:
54794  <pre><code>
54795 var reader = new Roo.ReaderLayout();
54796 var CP = Roo.ContentPanel;  // shortcut for adding
54797
54798 reader.beginUpdate();
54799 reader.add("north", new CP("north", "North"));
54800 reader.add("west", new CP("west", {title: "West"}));
54801 reader.add("east", new CP("east", {title: "East"}));
54802
54803 reader.regions.listView.add(new CP("listView", "List"));
54804 reader.regions.preview.add(new CP("preview", "Preview"));
54805 reader.endUpdate();
54806 </code></pre>
54807 * @constructor
54808 * Create a new ReaderLayout
54809 * @param {Object} config Configuration options
54810 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54811 * document.body if omitted)
54812 */
54813 Roo.ReaderLayout = function(config, renderTo){
54814     var c = config || {size:{}};
54815     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54816         north: c.north !== false ? Roo.apply({
54817             split:false,
54818             initialSize: 32,
54819             titlebar: false
54820         }, c.north) : false,
54821         west: c.west !== false ? Roo.apply({
54822             split:true,
54823             initialSize: 200,
54824             minSize: 175,
54825             maxSize: 400,
54826             titlebar: true,
54827             collapsible: true,
54828             animate: true,
54829             margins:{left:5,right:0,bottom:5,top:5},
54830             cmargins:{left:5,right:5,bottom:5,top:5}
54831         }, c.west) : false,
54832         east: c.east !== false ? Roo.apply({
54833             split:true,
54834             initialSize: 200,
54835             minSize: 175,
54836             maxSize: 400,
54837             titlebar: true,
54838             collapsible: true,
54839             animate: true,
54840             margins:{left:0,right:5,bottom:5,top:5},
54841             cmargins:{left:5,right:5,bottom:5,top:5}
54842         }, c.east) : false,
54843         center: Roo.apply({
54844             tabPosition: 'top',
54845             autoScroll:false,
54846             closeOnTab: true,
54847             titlebar:false,
54848             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54849         }, c.center)
54850     });
54851
54852     this.el.addClass('x-reader');
54853
54854     this.beginUpdate();
54855
54856     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54857         south: c.preview !== false ? Roo.apply({
54858             split:true,
54859             initialSize: 200,
54860             minSize: 100,
54861             autoScroll:true,
54862             collapsible:true,
54863             titlebar: true,
54864             cmargins:{top:5,left:0, right:0, bottom:0}
54865         }, c.preview) : false,
54866         center: Roo.apply({
54867             autoScroll:false,
54868             titlebar:false,
54869             minHeight:200
54870         }, c.listView)
54871     });
54872     this.add('center', new Roo.NestedLayoutPanel(inner,
54873             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54874
54875     this.endUpdate();
54876
54877     this.regions.preview = inner.getRegion('south');
54878     this.regions.listView = inner.getRegion('center');
54879 };
54880
54881 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54882  * Based on:
54883  * Ext JS Library 1.1.1
54884  * Copyright(c) 2006-2007, Ext JS, LLC.
54885  *
54886  * Originally Released Under LGPL - original licence link has changed is not relivant.
54887  *
54888  * Fork - LGPL
54889  * <script type="text/javascript">
54890  */
54891  
54892 /**
54893  * @class Roo.grid.Grid
54894  * @extends Roo.util.Observable
54895  * This class represents the primary interface of a component based grid control.
54896  * <br><br>Usage:<pre><code>
54897  var grid = new Roo.grid.Grid("my-container-id", {
54898      ds: myDataStore,
54899      cm: myColModel,
54900      selModel: mySelectionModel,
54901      autoSizeColumns: true,
54902      monitorWindowResize: false,
54903      trackMouseOver: true
54904  });
54905  // set any options
54906  grid.render();
54907  * </code></pre>
54908  * <b>Common Problems:</b><br/>
54909  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54910  * element will correct this<br/>
54911  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54912  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54913  * are unpredictable.<br/>
54914  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54915  * grid to calculate dimensions/offsets.<br/>
54916   * @constructor
54917  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54918  * The container MUST have some type of size defined for the grid to fill. The container will be
54919  * automatically set to position relative if it isn't already.
54920  * @param {Object} config A config object that sets properties on this grid.
54921  */
54922 Roo.grid.Grid = function(container, config){
54923         // initialize the container
54924         this.container = Roo.get(container);
54925         this.container.update("");
54926         this.container.setStyle("overflow", "hidden");
54927     this.container.addClass('x-grid-container');
54928
54929     this.id = this.container.id;
54930
54931     Roo.apply(this, config);
54932     // check and correct shorthanded configs
54933     if(this.ds){
54934         this.dataSource = this.ds;
54935         delete this.ds;
54936     }
54937     if(this.cm){
54938         this.colModel = this.cm;
54939         delete this.cm;
54940     }
54941     if(this.sm){
54942         this.selModel = this.sm;
54943         delete this.sm;
54944     }
54945
54946     if (this.selModel) {
54947         this.selModel = Roo.factory(this.selModel, Roo.grid);
54948         this.sm = this.selModel;
54949         this.sm.xmodule = this.xmodule || false;
54950     }
54951     if (typeof(this.colModel.config) == 'undefined') {
54952         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54953         this.cm = this.colModel;
54954         this.cm.xmodule = this.xmodule || false;
54955     }
54956     if (this.dataSource) {
54957         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54958         this.ds = this.dataSource;
54959         this.ds.xmodule = this.xmodule || false;
54960          
54961     }
54962     
54963     
54964     
54965     if(this.width){
54966         this.container.setWidth(this.width);
54967     }
54968
54969     if(this.height){
54970         this.container.setHeight(this.height);
54971     }
54972     /** @private */
54973         this.addEvents({
54974         // raw events
54975         /**
54976          * @event click
54977          * The raw click event for the entire grid.
54978          * @param {Roo.EventObject} e
54979          */
54980         "click" : true,
54981         /**
54982          * @event dblclick
54983          * The raw dblclick event for the entire grid.
54984          * @param {Roo.EventObject} e
54985          */
54986         "dblclick" : true,
54987         /**
54988          * @event contextmenu
54989          * The raw contextmenu event for the entire grid.
54990          * @param {Roo.EventObject} e
54991          */
54992         "contextmenu" : true,
54993         /**
54994          * @event mousedown
54995          * The raw mousedown event for the entire grid.
54996          * @param {Roo.EventObject} e
54997          */
54998         "mousedown" : true,
54999         /**
55000          * @event mouseup
55001          * The raw mouseup event for the entire grid.
55002          * @param {Roo.EventObject} e
55003          */
55004         "mouseup" : true,
55005         /**
55006          * @event mouseover
55007          * The raw mouseover event for the entire grid.
55008          * @param {Roo.EventObject} e
55009          */
55010         "mouseover" : true,
55011         /**
55012          * @event mouseout
55013          * The raw mouseout event for the entire grid.
55014          * @param {Roo.EventObject} e
55015          */
55016         "mouseout" : true,
55017         /**
55018          * @event keypress
55019          * The raw keypress event for the entire grid.
55020          * @param {Roo.EventObject} e
55021          */
55022         "keypress" : true,
55023         /**
55024          * @event keydown
55025          * The raw keydown event for the entire grid.
55026          * @param {Roo.EventObject} e
55027          */
55028         "keydown" : true,
55029
55030         // custom events
55031
55032         /**
55033          * @event cellclick
55034          * Fires when a cell is clicked
55035          * @param {Grid} this
55036          * @param {Number} rowIndex
55037          * @param {Number} columnIndex
55038          * @param {Roo.EventObject} e
55039          */
55040         "cellclick" : true,
55041         /**
55042          * @event celldblclick
55043          * Fires when a cell is double clicked
55044          * @param {Grid} this
55045          * @param {Number} rowIndex
55046          * @param {Number} columnIndex
55047          * @param {Roo.EventObject} e
55048          */
55049         "celldblclick" : true,
55050         /**
55051          * @event rowclick
55052          * Fires when a row is clicked
55053          * @param {Grid} this
55054          * @param {Number} rowIndex
55055          * @param {Roo.EventObject} e
55056          */
55057         "rowclick" : true,
55058         /**
55059          * @event rowdblclick
55060          * Fires when a row is double clicked
55061          * @param {Grid} this
55062          * @param {Number} rowIndex
55063          * @param {Roo.EventObject} e
55064          */
55065         "rowdblclick" : true,
55066         /**
55067          * @event headerclick
55068          * Fires when a header is clicked
55069          * @param {Grid} this
55070          * @param {Number} columnIndex
55071          * @param {Roo.EventObject} e
55072          */
55073         "headerclick" : true,
55074         /**
55075          * @event headerdblclick
55076          * Fires when a header cell is double clicked
55077          * @param {Grid} this
55078          * @param {Number} columnIndex
55079          * @param {Roo.EventObject} e
55080          */
55081         "headerdblclick" : true,
55082         /**
55083          * @event rowcontextmenu
55084          * Fires when a row is right clicked
55085          * @param {Grid} this
55086          * @param {Number} rowIndex
55087          * @param {Roo.EventObject} e
55088          */
55089         "rowcontextmenu" : true,
55090         /**
55091          * @event cellcontextmenu
55092          * Fires when a cell is right clicked
55093          * @param {Grid} this
55094          * @param {Number} rowIndex
55095          * @param {Number} cellIndex
55096          * @param {Roo.EventObject} e
55097          */
55098          "cellcontextmenu" : true,
55099         /**
55100          * @event headercontextmenu
55101          * Fires when a header is right clicked
55102          * @param {Grid} this
55103          * @param {Number} columnIndex
55104          * @param {Roo.EventObject} e
55105          */
55106         "headercontextmenu" : true,
55107         /**
55108          * @event bodyscroll
55109          * Fires when the body element is scrolled
55110          * @param {Number} scrollLeft
55111          * @param {Number} scrollTop
55112          */
55113         "bodyscroll" : true,
55114         /**
55115          * @event columnresize
55116          * Fires when the user resizes a column
55117          * @param {Number} columnIndex
55118          * @param {Number} newSize
55119          */
55120         "columnresize" : true,
55121         /**
55122          * @event columnmove
55123          * Fires when the user moves a column
55124          * @param {Number} oldIndex
55125          * @param {Number} newIndex
55126          */
55127         "columnmove" : true,
55128         /**
55129          * @event startdrag
55130          * Fires when row(s) start being dragged
55131          * @param {Grid} this
55132          * @param {Roo.GridDD} dd The drag drop object
55133          * @param {event} e The raw browser event
55134          */
55135         "startdrag" : true,
55136         /**
55137          * @event enddrag
55138          * Fires when a drag operation is complete
55139          * @param {Grid} this
55140          * @param {Roo.GridDD} dd The drag drop object
55141          * @param {event} e The raw browser event
55142          */
55143         "enddrag" : true,
55144         /**
55145          * @event dragdrop
55146          * Fires when dragged row(s) are dropped on a valid DD target
55147          * @param {Grid} this
55148          * @param {Roo.GridDD} dd The drag drop object
55149          * @param {String} targetId The target drag drop object
55150          * @param {event} e The raw browser event
55151          */
55152         "dragdrop" : true,
55153         /**
55154          * @event dragover
55155          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55156          * @param {Grid} this
55157          * @param {Roo.GridDD} dd The drag drop object
55158          * @param {String} targetId The target drag drop object
55159          * @param {event} e The raw browser event
55160          */
55161         "dragover" : true,
55162         /**
55163          * @event dragenter
55164          *  Fires when the dragged row(s) first cross another DD target while being dragged
55165          * @param {Grid} this
55166          * @param {Roo.GridDD} dd The drag drop object
55167          * @param {String} targetId The target drag drop object
55168          * @param {event} e The raw browser event
55169          */
55170         "dragenter" : true,
55171         /**
55172          * @event dragout
55173          * Fires when the dragged row(s) leave another DD target while being dragged
55174          * @param {Grid} this
55175          * @param {Roo.GridDD} dd The drag drop object
55176          * @param {String} targetId The target drag drop object
55177          * @param {event} e The raw browser event
55178          */
55179         "dragout" : true,
55180         /**
55181          * @event rowclass
55182          * Fires when a row is rendered, so you can change add a style to it.
55183          * @param {GridView} gridview   The grid view
55184          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55185          */
55186         'rowclass' : true,
55187
55188         /**
55189          * @event render
55190          * Fires when the grid is rendered
55191          * @param {Grid} grid
55192          */
55193         'render' : true
55194     });
55195
55196     Roo.grid.Grid.superclass.constructor.call(this);
55197 };
55198 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55199     
55200     /**
55201      * @cfg {String} ddGroup - drag drop group.
55202      */
55203
55204     /**
55205      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55206      */
55207     minColumnWidth : 25,
55208
55209     /**
55210      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55211      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55212      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55213      */
55214     autoSizeColumns : false,
55215
55216     /**
55217      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55218      */
55219     autoSizeHeaders : true,
55220
55221     /**
55222      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55223      */
55224     monitorWindowResize : true,
55225
55226     /**
55227      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55228      * rows measured to get a columns size. Default is 0 (all rows).
55229      */
55230     maxRowsToMeasure : 0,
55231
55232     /**
55233      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55234      */
55235     trackMouseOver : true,
55236
55237     /**
55238     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55239     */
55240     
55241     /**
55242     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55243     */
55244     enableDragDrop : false,
55245     
55246     /**
55247     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55248     */
55249     enableColumnMove : true,
55250     
55251     /**
55252     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55253     */
55254     enableColumnHide : true,
55255     
55256     /**
55257     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55258     */
55259     enableRowHeightSync : false,
55260     
55261     /**
55262     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55263     */
55264     stripeRows : true,
55265     
55266     /**
55267     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55268     */
55269     autoHeight : false,
55270
55271     /**
55272      * @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.
55273      */
55274     autoExpandColumn : false,
55275
55276     /**
55277     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55278     * Default is 50.
55279     */
55280     autoExpandMin : 50,
55281
55282     /**
55283     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55284     */
55285     autoExpandMax : 1000,
55286
55287     /**
55288     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55289     */
55290     view : null,
55291
55292     /**
55293     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55294     */
55295     loadMask : false,
55296     /**
55297     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55298     */
55299     dropTarget: false,
55300     
55301    
55302     
55303     // private
55304     rendered : false,
55305
55306     /**
55307     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55308     * of a fixed width. Default is false.
55309     */
55310     /**
55311     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55312     */
55313     /**
55314      * Called once after all setup has been completed and the grid is ready to be rendered.
55315      * @return {Roo.grid.Grid} this
55316      */
55317     render : function()
55318     {
55319         var c = this.container;
55320         // try to detect autoHeight/width mode
55321         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55322             this.autoHeight = true;
55323         }
55324         var view = this.getView();
55325         view.init(this);
55326
55327         c.on("click", this.onClick, this);
55328         c.on("dblclick", this.onDblClick, this);
55329         c.on("contextmenu", this.onContextMenu, this);
55330         c.on("keydown", this.onKeyDown, this);
55331         if (Roo.isTouch) {
55332             c.on("touchstart", this.onTouchStart, this);
55333         }
55334
55335         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55336
55337         this.getSelectionModel().init(this);
55338
55339         view.render();
55340
55341         if(this.loadMask){
55342             this.loadMask = new Roo.LoadMask(this.container,
55343                     Roo.apply({store:this.dataSource}, this.loadMask));
55344         }
55345         
55346         
55347         if (this.toolbar && this.toolbar.xtype) {
55348             this.toolbar.container = this.getView().getHeaderPanel(true);
55349             this.toolbar = new Roo.Toolbar(this.toolbar);
55350         }
55351         if (this.footer && this.footer.xtype) {
55352             this.footer.dataSource = this.getDataSource();
55353             this.footer.container = this.getView().getFooterPanel(true);
55354             this.footer = Roo.factory(this.footer, Roo);
55355         }
55356         if (this.dropTarget && this.dropTarget.xtype) {
55357             delete this.dropTarget.xtype;
55358             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55359         }
55360         
55361         
55362         this.rendered = true;
55363         this.fireEvent('render', this);
55364         return this;
55365     },
55366
55367         /**
55368          * Reconfigures the grid to use a different Store and Column Model.
55369          * The View will be bound to the new objects and refreshed.
55370          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55371          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55372          */
55373     reconfigure : function(dataSource, colModel){
55374         if(this.loadMask){
55375             this.loadMask.destroy();
55376             this.loadMask = new Roo.LoadMask(this.container,
55377                     Roo.apply({store:dataSource}, this.loadMask));
55378         }
55379         this.view.bind(dataSource, colModel);
55380         this.dataSource = dataSource;
55381         this.colModel = colModel;
55382         this.view.refresh(true);
55383     },
55384
55385     // private
55386     onKeyDown : function(e){
55387         this.fireEvent("keydown", e);
55388     },
55389
55390     /**
55391      * Destroy this grid.
55392      * @param {Boolean} removeEl True to remove the element
55393      */
55394     destroy : function(removeEl, keepListeners){
55395         if(this.loadMask){
55396             this.loadMask.destroy();
55397         }
55398         var c = this.container;
55399         c.removeAllListeners();
55400         this.view.destroy();
55401         this.colModel.purgeListeners();
55402         if(!keepListeners){
55403             this.purgeListeners();
55404         }
55405         c.update("");
55406         if(removeEl === true){
55407             c.remove();
55408         }
55409     },
55410
55411     // private
55412     processEvent : function(name, e){
55413         // does this fire select???
55414         //Roo.log('grid:processEvent '  + name);
55415         
55416         if (name != 'touchstart' ) {
55417             this.fireEvent(name, e);    
55418         }
55419         
55420         var t = e.getTarget();
55421         var v = this.view;
55422         var header = v.findHeaderIndex(t);
55423         if(header !== false){
55424             var ename = name == 'touchstart' ? 'click' : name;
55425              
55426             this.fireEvent("header" + ename, this, header, e);
55427         }else{
55428             var row = v.findRowIndex(t);
55429             var cell = v.findCellIndex(t);
55430             if (name == 'touchstart') {
55431                 // first touch is always a click.
55432                 // hopefull this happens after selection is updated.?
55433                 name = false;
55434                 
55435                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55436                     var cs = this.selModel.getSelectedCell();
55437                     if (row == cs[0] && cell == cs[1]){
55438                         name = 'dblclick';
55439                     }
55440                 }
55441                 if (typeof(this.selModel.getSelections) != 'undefined') {
55442                     var cs = this.selModel.getSelections();
55443                     var ds = this.dataSource;
55444                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55445                         name = 'dblclick';
55446                     }
55447                 }
55448                 if (!name) {
55449                     return;
55450                 }
55451             }
55452             
55453             
55454             if(row !== false){
55455                 this.fireEvent("row" + name, this, row, e);
55456                 if(cell !== false){
55457                     this.fireEvent("cell" + name, this, row, cell, e);
55458                 }
55459             }
55460         }
55461     },
55462
55463     // private
55464     onClick : function(e){
55465         this.processEvent("click", e);
55466     },
55467    // private
55468     onTouchStart : function(e){
55469         this.processEvent("touchstart", e);
55470     },
55471
55472     // private
55473     onContextMenu : function(e, t){
55474         this.processEvent("contextmenu", e);
55475     },
55476
55477     // private
55478     onDblClick : function(e){
55479         this.processEvent("dblclick", e);
55480     },
55481
55482     // private
55483     walkCells : function(row, col, step, fn, scope){
55484         var cm = this.colModel, clen = cm.getColumnCount();
55485         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55486         if(step < 0){
55487             if(col < 0){
55488                 row--;
55489                 first = false;
55490             }
55491             while(row >= 0){
55492                 if(!first){
55493                     col = clen-1;
55494                 }
55495                 first = false;
55496                 while(col >= 0){
55497                     if(fn.call(scope || this, row, col, cm) === true){
55498                         return [row, col];
55499                     }
55500                     col--;
55501                 }
55502                 row--;
55503             }
55504         } else {
55505             if(col >= clen){
55506                 row++;
55507                 first = false;
55508             }
55509             while(row < rlen){
55510                 if(!first){
55511                     col = 0;
55512                 }
55513                 first = false;
55514                 while(col < clen){
55515                     if(fn.call(scope || this, row, col, cm) === true){
55516                         return [row, col];
55517                     }
55518                     col++;
55519                 }
55520                 row++;
55521             }
55522         }
55523         return null;
55524     },
55525
55526     // private
55527     getSelections : function(){
55528         return this.selModel.getSelections();
55529     },
55530
55531     /**
55532      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55533      * but if manual update is required this method will initiate it.
55534      */
55535     autoSize : function(){
55536         if(this.rendered){
55537             this.view.layout();
55538             if(this.view.adjustForScroll){
55539                 this.view.adjustForScroll();
55540             }
55541         }
55542     },
55543
55544     /**
55545      * Returns the grid's underlying element.
55546      * @return {Element} The element
55547      */
55548     getGridEl : function(){
55549         return this.container;
55550     },
55551
55552     // private for compatibility, overridden by editor grid
55553     stopEditing : function(){},
55554
55555     /**
55556      * Returns the grid's SelectionModel.
55557      * @return {SelectionModel}
55558      */
55559     getSelectionModel : function(){
55560         if(!this.selModel){
55561             this.selModel = new Roo.grid.RowSelectionModel();
55562         }
55563         return this.selModel;
55564     },
55565
55566     /**
55567      * Returns the grid's DataSource.
55568      * @return {DataSource}
55569      */
55570     getDataSource : function(){
55571         return this.dataSource;
55572     },
55573
55574     /**
55575      * Returns the grid's ColumnModel.
55576      * @return {ColumnModel}
55577      */
55578     getColumnModel : function(){
55579         return this.colModel;
55580     },
55581
55582     /**
55583      * Returns the grid's GridView object.
55584      * @return {GridView}
55585      */
55586     getView : function(){
55587         if(!this.view){
55588             this.view = new Roo.grid.GridView(this.viewConfig);
55589         }
55590         return this.view;
55591     },
55592     /**
55593      * Called to get grid's drag proxy text, by default returns this.ddText.
55594      * @return {String}
55595      */
55596     getDragDropText : function(){
55597         var count = this.selModel.getCount();
55598         return String.format(this.ddText, count, count == 1 ? '' : 's');
55599     }
55600 });
55601 /**
55602  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55603  * %0 is replaced with the number of selected rows.
55604  * @type String
55605  */
55606 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55607  * Based on:
55608  * Ext JS Library 1.1.1
55609  * Copyright(c) 2006-2007, Ext JS, LLC.
55610  *
55611  * Originally Released Under LGPL - original licence link has changed is not relivant.
55612  *
55613  * Fork - LGPL
55614  * <script type="text/javascript">
55615  */
55616  
55617 Roo.grid.AbstractGridView = function(){
55618         this.grid = null;
55619         
55620         this.events = {
55621             "beforerowremoved" : true,
55622             "beforerowsinserted" : true,
55623             "beforerefresh" : true,
55624             "rowremoved" : true,
55625             "rowsinserted" : true,
55626             "rowupdated" : true,
55627             "refresh" : true
55628         };
55629     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55630 };
55631
55632 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55633     rowClass : "x-grid-row",
55634     cellClass : "x-grid-cell",
55635     tdClass : "x-grid-td",
55636     hdClass : "x-grid-hd",
55637     splitClass : "x-grid-hd-split",
55638     
55639     init: function(grid){
55640         this.grid = grid;
55641                 var cid = this.grid.getGridEl().id;
55642         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55643         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55644         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55645         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55646         },
55647         
55648     getColumnRenderers : function(){
55649         var renderers = [];
55650         var cm = this.grid.colModel;
55651         var colCount = cm.getColumnCount();
55652         for(var i = 0; i < colCount; i++){
55653             renderers[i] = cm.getRenderer(i);
55654         }
55655         return renderers;
55656     },
55657     
55658     getColumnIds : function(){
55659         var ids = [];
55660         var cm = this.grid.colModel;
55661         var colCount = cm.getColumnCount();
55662         for(var i = 0; i < colCount; i++){
55663             ids[i] = cm.getColumnId(i);
55664         }
55665         return ids;
55666     },
55667     
55668     getDataIndexes : function(){
55669         if(!this.indexMap){
55670             this.indexMap = this.buildIndexMap();
55671         }
55672         return this.indexMap.colToData;
55673     },
55674     
55675     getColumnIndexByDataIndex : function(dataIndex){
55676         if(!this.indexMap){
55677             this.indexMap = this.buildIndexMap();
55678         }
55679         return this.indexMap.dataToCol[dataIndex];
55680     },
55681     
55682     /**
55683      * Set a css style for a column dynamically. 
55684      * @param {Number} colIndex The index of the column
55685      * @param {String} name The css property name
55686      * @param {String} value The css value
55687      */
55688     setCSSStyle : function(colIndex, name, value){
55689         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55690         Roo.util.CSS.updateRule(selector, name, value);
55691     },
55692     
55693     generateRules : function(cm){
55694         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55695         Roo.util.CSS.removeStyleSheet(rulesId);
55696         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55697             var cid = cm.getColumnId(i);
55698             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55699                          this.tdSelector, cid, " {\n}\n",
55700                          this.hdSelector, cid, " {\n}\n",
55701                          this.splitSelector, cid, " {\n}\n");
55702         }
55703         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55704     }
55705 });/*
55706  * Based on:
55707  * Ext JS Library 1.1.1
55708  * Copyright(c) 2006-2007, Ext JS, LLC.
55709  *
55710  * Originally Released Under LGPL - original licence link has changed is not relivant.
55711  *
55712  * Fork - LGPL
55713  * <script type="text/javascript">
55714  */
55715
55716 // private
55717 // This is a support class used internally by the Grid components
55718 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55719     this.grid = grid;
55720     this.view = grid.getView();
55721     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55722     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55723     if(hd2){
55724         this.setHandleElId(Roo.id(hd));
55725         this.setOuterHandleElId(Roo.id(hd2));
55726     }
55727     this.scroll = false;
55728 };
55729 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55730     maxDragWidth: 120,
55731     getDragData : function(e){
55732         var t = Roo.lib.Event.getTarget(e);
55733         var h = this.view.findHeaderCell(t);
55734         if(h){
55735             return {ddel: h.firstChild, header:h};
55736         }
55737         return false;
55738     },
55739
55740     onInitDrag : function(e){
55741         this.view.headersDisabled = true;
55742         var clone = this.dragData.ddel.cloneNode(true);
55743         clone.id = Roo.id();
55744         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55745         this.proxy.update(clone);
55746         return true;
55747     },
55748
55749     afterValidDrop : function(){
55750         var v = this.view;
55751         setTimeout(function(){
55752             v.headersDisabled = false;
55753         }, 50);
55754     },
55755
55756     afterInvalidDrop : function(){
55757         var v = this.view;
55758         setTimeout(function(){
55759             v.headersDisabled = false;
55760         }, 50);
55761     }
55762 });
55763 /*
55764  * Based on:
55765  * Ext JS Library 1.1.1
55766  * Copyright(c) 2006-2007, Ext JS, LLC.
55767  *
55768  * Originally Released Under LGPL - original licence link has changed is not relivant.
55769  *
55770  * Fork - LGPL
55771  * <script type="text/javascript">
55772  */
55773 // private
55774 // This is a support class used internally by the Grid components
55775 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55776     this.grid = grid;
55777     this.view = grid.getView();
55778     // split the proxies so they don't interfere with mouse events
55779     this.proxyTop = Roo.DomHelper.append(document.body, {
55780         cls:"col-move-top", html:"&#160;"
55781     }, true);
55782     this.proxyBottom = Roo.DomHelper.append(document.body, {
55783         cls:"col-move-bottom", html:"&#160;"
55784     }, true);
55785     this.proxyTop.hide = this.proxyBottom.hide = function(){
55786         this.setLeftTop(-100,-100);
55787         this.setStyle("visibility", "hidden");
55788     };
55789     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55790     // temporarily disabled
55791     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55792     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55793 };
55794 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55795     proxyOffsets : [-4, -9],
55796     fly: Roo.Element.fly,
55797
55798     getTargetFromEvent : function(e){
55799         var t = Roo.lib.Event.getTarget(e);
55800         var cindex = this.view.findCellIndex(t);
55801         if(cindex !== false){
55802             return this.view.getHeaderCell(cindex);
55803         }
55804         return null;
55805     },
55806
55807     nextVisible : function(h){
55808         var v = this.view, cm = this.grid.colModel;
55809         h = h.nextSibling;
55810         while(h){
55811             if(!cm.isHidden(v.getCellIndex(h))){
55812                 return h;
55813             }
55814             h = h.nextSibling;
55815         }
55816         return null;
55817     },
55818
55819     prevVisible : function(h){
55820         var v = this.view, cm = this.grid.colModel;
55821         h = h.prevSibling;
55822         while(h){
55823             if(!cm.isHidden(v.getCellIndex(h))){
55824                 return h;
55825             }
55826             h = h.prevSibling;
55827         }
55828         return null;
55829     },
55830
55831     positionIndicator : function(h, n, e){
55832         var x = Roo.lib.Event.getPageX(e);
55833         var r = Roo.lib.Dom.getRegion(n.firstChild);
55834         var px, pt, py = r.top + this.proxyOffsets[1];
55835         if((r.right - x) <= (r.right-r.left)/2){
55836             px = r.right+this.view.borderWidth;
55837             pt = "after";
55838         }else{
55839             px = r.left;
55840             pt = "before";
55841         }
55842         var oldIndex = this.view.getCellIndex(h);
55843         var newIndex = this.view.getCellIndex(n);
55844
55845         if(this.grid.colModel.isFixed(newIndex)){
55846             return false;
55847         }
55848
55849         var locked = this.grid.colModel.isLocked(newIndex);
55850
55851         if(pt == "after"){
55852             newIndex++;
55853         }
55854         if(oldIndex < newIndex){
55855             newIndex--;
55856         }
55857         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55858             return false;
55859         }
55860         px +=  this.proxyOffsets[0];
55861         this.proxyTop.setLeftTop(px, py);
55862         this.proxyTop.show();
55863         if(!this.bottomOffset){
55864             this.bottomOffset = this.view.mainHd.getHeight();
55865         }
55866         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55867         this.proxyBottom.show();
55868         return pt;
55869     },
55870
55871     onNodeEnter : function(n, dd, e, data){
55872         if(data.header != n){
55873             this.positionIndicator(data.header, n, e);
55874         }
55875     },
55876
55877     onNodeOver : function(n, dd, e, data){
55878         var result = false;
55879         if(data.header != n){
55880             result = this.positionIndicator(data.header, n, e);
55881         }
55882         if(!result){
55883             this.proxyTop.hide();
55884             this.proxyBottom.hide();
55885         }
55886         return result ? this.dropAllowed : this.dropNotAllowed;
55887     },
55888
55889     onNodeOut : function(n, dd, e, data){
55890         this.proxyTop.hide();
55891         this.proxyBottom.hide();
55892     },
55893
55894     onNodeDrop : function(n, dd, e, data){
55895         var h = data.header;
55896         if(h != n){
55897             var cm = this.grid.colModel;
55898             var x = Roo.lib.Event.getPageX(e);
55899             var r = Roo.lib.Dom.getRegion(n.firstChild);
55900             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55901             var oldIndex = this.view.getCellIndex(h);
55902             var newIndex = this.view.getCellIndex(n);
55903             var locked = cm.isLocked(newIndex);
55904             if(pt == "after"){
55905                 newIndex++;
55906             }
55907             if(oldIndex < newIndex){
55908                 newIndex--;
55909             }
55910             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55911                 return false;
55912             }
55913             cm.setLocked(oldIndex, locked, true);
55914             cm.moveColumn(oldIndex, newIndex);
55915             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55916             return true;
55917         }
55918         return false;
55919     }
55920 });
55921 /*
55922  * Based on:
55923  * Ext JS Library 1.1.1
55924  * Copyright(c) 2006-2007, Ext JS, LLC.
55925  *
55926  * Originally Released Under LGPL - original licence link has changed is not relivant.
55927  *
55928  * Fork - LGPL
55929  * <script type="text/javascript">
55930  */
55931   
55932 /**
55933  * @class Roo.grid.GridView
55934  * @extends Roo.util.Observable
55935  *
55936  * @constructor
55937  * @param {Object} config
55938  */
55939 Roo.grid.GridView = function(config){
55940     Roo.grid.GridView.superclass.constructor.call(this);
55941     this.el = null;
55942
55943     Roo.apply(this, config);
55944 };
55945
55946 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55947
55948     unselectable :  'unselectable="on"',
55949     unselectableCls :  'x-unselectable',
55950     
55951     
55952     rowClass : "x-grid-row",
55953
55954     cellClass : "x-grid-col",
55955
55956     tdClass : "x-grid-td",
55957
55958     hdClass : "x-grid-hd",
55959
55960     splitClass : "x-grid-split",
55961
55962     sortClasses : ["sort-asc", "sort-desc"],
55963
55964     enableMoveAnim : false,
55965
55966     hlColor: "C3DAF9",
55967
55968     dh : Roo.DomHelper,
55969
55970     fly : Roo.Element.fly,
55971
55972     css : Roo.util.CSS,
55973
55974     borderWidth: 1,
55975
55976     splitOffset: 3,
55977
55978     scrollIncrement : 22,
55979
55980     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55981
55982     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55983
55984     bind : function(ds, cm){
55985         if(this.ds){
55986             this.ds.un("load", this.onLoad, this);
55987             this.ds.un("datachanged", this.onDataChange, this);
55988             this.ds.un("add", this.onAdd, this);
55989             this.ds.un("remove", this.onRemove, this);
55990             this.ds.un("update", this.onUpdate, this);
55991             this.ds.un("clear", this.onClear, this);
55992         }
55993         if(ds){
55994             ds.on("load", this.onLoad, this);
55995             ds.on("datachanged", this.onDataChange, this);
55996             ds.on("add", this.onAdd, this);
55997             ds.on("remove", this.onRemove, this);
55998             ds.on("update", this.onUpdate, this);
55999             ds.on("clear", this.onClear, this);
56000         }
56001         this.ds = ds;
56002
56003         if(this.cm){
56004             this.cm.un("widthchange", this.onColWidthChange, this);
56005             this.cm.un("headerchange", this.onHeaderChange, this);
56006             this.cm.un("hiddenchange", this.onHiddenChange, this);
56007             this.cm.un("columnmoved", this.onColumnMove, this);
56008             this.cm.un("columnlockchange", this.onColumnLock, this);
56009         }
56010         if(cm){
56011             this.generateRules(cm);
56012             cm.on("widthchange", this.onColWidthChange, this);
56013             cm.on("headerchange", this.onHeaderChange, this);
56014             cm.on("hiddenchange", this.onHiddenChange, this);
56015             cm.on("columnmoved", this.onColumnMove, this);
56016             cm.on("columnlockchange", this.onColumnLock, this);
56017         }
56018         this.cm = cm;
56019     },
56020
56021     init: function(grid){
56022         Roo.grid.GridView.superclass.init.call(this, grid);
56023
56024         this.bind(grid.dataSource, grid.colModel);
56025
56026         grid.on("headerclick", this.handleHeaderClick, this);
56027
56028         if(grid.trackMouseOver){
56029             grid.on("mouseover", this.onRowOver, this);
56030             grid.on("mouseout", this.onRowOut, this);
56031         }
56032         grid.cancelTextSelection = function(){};
56033         this.gridId = grid.id;
56034
56035         var tpls = this.templates || {};
56036
56037         if(!tpls.master){
56038             tpls.master = new Roo.Template(
56039                '<div class="x-grid" hidefocus="true">',
56040                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56041                   '<div class="x-grid-topbar"></div>',
56042                   '<div class="x-grid-scroller"><div></div></div>',
56043                   '<div class="x-grid-locked">',
56044                       '<div class="x-grid-header">{lockedHeader}</div>',
56045                       '<div class="x-grid-body">{lockedBody}</div>',
56046                   "</div>",
56047                   '<div class="x-grid-viewport">',
56048                       '<div class="x-grid-header">{header}</div>',
56049                       '<div class="x-grid-body">{body}</div>',
56050                   "</div>",
56051                   '<div class="x-grid-bottombar"></div>',
56052                  
56053                   '<div class="x-grid-resize-proxy">&#160;</div>',
56054                "</div>"
56055             );
56056             tpls.master.disableformats = true;
56057         }
56058
56059         if(!tpls.header){
56060             tpls.header = new Roo.Template(
56061                '<table border="0" cellspacing="0" cellpadding="0">',
56062                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56063                "</table>{splits}"
56064             );
56065             tpls.header.disableformats = true;
56066         }
56067         tpls.header.compile();
56068
56069         if(!tpls.hcell){
56070             tpls.hcell = new Roo.Template(
56071                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56072                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56073                 "</div></td>"
56074              );
56075              tpls.hcell.disableFormats = true;
56076         }
56077         tpls.hcell.compile();
56078
56079         if(!tpls.hsplit){
56080             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56081                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56082             tpls.hsplit.disableFormats = true;
56083         }
56084         tpls.hsplit.compile();
56085
56086         if(!tpls.body){
56087             tpls.body = new Roo.Template(
56088                '<table border="0" cellspacing="0" cellpadding="0">',
56089                "<tbody>{rows}</tbody>",
56090                "</table>"
56091             );
56092             tpls.body.disableFormats = true;
56093         }
56094         tpls.body.compile();
56095
56096         if(!tpls.row){
56097             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56098             tpls.row.disableFormats = true;
56099         }
56100         tpls.row.compile();
56101
56102         if(!tpls.cell){
56103             tpls.cell = new Roo.Template(
56104                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56105                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56106                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56107                 "</td>"
56108             );
56109             tpls.cell.disableFormats = true;
56110         }
56111         tpls.cell.compile();
56112
56113         this.templates = tpls;
56114     },
56115
56116     // remap these for backwards compat
56117     onColWidthChange : function(){
56118         this.updateColumns.apply(this, arguments);
56119     },
56120     onHeaderChange : function(){
56121         this.updateHeaders.apply(this, arguments);
56122     }, 
56123     onHiddenChange : function(){
56124         this.handleHiddenChange.apply(this, arguments);
56125     },
56126     onColumnMove : function(){
56127         this.handleColumnMove.apply(this, arguments);
56128     },
56129     onColumnLock : function(){
56130         this.handleLockChange.apply(this, arguments);
56131     },
56132
56133     onDataChange : function(){
56134         this.refresh();
56135         this.updateHeaderSortState();
56136     },
56137
56138     onClear : function(){
56139         this.refresh();
56140     },
56141
56142     onUpdate : function(ds, record){
56143         this.refreshRow(record);
56144     },
56145
56146     refreshRow : function(record){
56147         var ds = this.ds, index;
56148         if(typeof record == 'number'){
56149             index = record;
56150             record = ds.getAt(index);
56151         }else{
56152             index = ds.indexOf(record);
56153         }
56154         this.insertRows(ds, index, index, true);
56155         this.onRemove(ds, record, index+1, true);
56156         this.syncRowHeights(index, index);
56157         this.layout();
56158         this.fireEvent("rowupdated", this, index, record);
56159     },
56160
56161     onAdd : function(ds, records, index){
56162         this.insertRows(ds, index, index + (records.length-1));
56163     },
56164
56165     onRemove : function(ds, record, index, isUpdate){
56166         if(isUpdate !== true){
56167             this.fireEvent("beforerowremoved", this, index, record);
56168         }
56169         var bt = this.getBodyTable(), lt = this.getLockedTable();
56170         if(bt.rows[index]){
56171             bt.firstChild.removeChild(bt.rows[index]);
56172         }
56173         if(lt.rows[index]){
56174             lt.firstChild.removeChild(lt.rows[index]);
56175         }
56176         if(isUpdate !== true){
56177             this.stripeRows(index);
56178             this.syncRowHeights(index, index);
56179             this.layout();
56180             this.fireEvent("rowremoved", this, index, record);
56181         }
56182     },
56183
56184     onLoad : function(){
56185         this.scrollToTop();
56186     },
56187
56188     /**
56189      * Scrolls the grid to the top
56190      */
56191     scrollToTop : function(){
56192         if(this.scroller){
56193             this.scroller.dom.scrollTop = 0;
56194             this.syncScroll();
56195         }
56196     },
56197
56198     /**
56199      * Gets a panel in the header of the grid that can be used for toolbars etc.
56200      * After modifying the contents of this panel a call to grid.autoSize() may be
56201      * required to register any changes in size.
56202      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56203      * @return Roo.Element
56204      */
56205     getHeaderPanel : function(doShow){
56206         if(doShow){
56207             this.headerPanel.show();
56208         }
56209         return this.headerPanel;
56210     },
56211
56212     /**
56213      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56214      * After modifying the contents of this panel a call to grid.autoSize() may be
56215      * required to register any changes in size.
56216      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56217      * @return Roo.Element
56218      */
56219     getFooterPanel : function(doShow){
56220         if(doShow){
56221             this.footerPanel.show();
56222         }
56223         return this.footerPanel;
56224     },
56225
56226     initElements : function(){
56227         var E = Roo.Element;
56228         var el = this.grid.getGridEl().dom.firstChild;
56229         var cs = el.childNodes;
56230
56231         this.el = new E(el);
56232         
56233          this.focusEl = new E(el.firstChild);
56234         this.focusEl.swallowEvent("click", true);
56235         
56236         this.headerPanel = new E(cs[1]);
56237         this.headerPanel.enableDisplayMode("block");
56238
56239         this.scroller = new E(cs[2]);
56240         this.scrollSizer = new E(this.scroller.dom.firstChild);
56241
56242         this.lockedWrap = new E(cs[3]);
56243         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56244         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56245
56246         this.mainWrap = new E(cs[4]);
56247         this.mainHd = new E(this.mainWrap.dom.firstChild);
56248         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56249
56250         this.footerPanel = new E(cs[5]);
56251         this.footerPanel.enableDisplayMode("block");
56252
56253         this.resizeProxy = new E(cs[6]);
56254
56255         this.headerSelector = String.format(
56256            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56257            this.lockedHd.id, this.mainHd.id
56258         );
56259
56260         this.splitterSelector = String.format(
56261            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56262            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56263         );
56264     },
56265     idToCssName : function(s)
56266     {
56267         return s.replace(/[^a-z0-9]+/ig, '-');
56268     },
56269
56270     getHeaderCell : function(index){
56271         return Roo.DomQuery.select(this.headerSelector)[index];
56272     },
56273
56274     getHeaderCellMeasure : function(index){
56275         return this.getHeaderCell(index).firstChild;
56276     },
56277
56278     getHeaderCellText : function(index){
56279         return this.getHeaderCell(index).firstChild.firstChild;
56280     },
56281
56282     getLockedTable : function(){
56283         return this.lockedBody.dom.firstChild;
56284     },
56285
56286     getBodyTable : function(){
56287         return this.mainBody.dom.firstChild;
56288     },
56289
56290     getLockedRow : function(index){
56291         return this.getLockedTable().rows[index];
56292     },
56293
56294     getRow : function(index){
56295         return this.getBodyTable().rows[index];
56296     },
56297
56298     getRowComposite : function(index){
56299         if(!this.rowEl){
56300             this.rowEl = new Roo.CompositeElementLite();
56301         }
56302         var els = [], lrow, mrow;
56303         if(lrow = this.getLockedRow(index)){
56304             els.push(lrow);
56305         }
56306         if(mrow = this.getRow(index)){
56307             els.push(mrow);
56308         }
56309         this.rowEl.elements = els;
56310         return this.rowEl;
56311     },
56312     /**
56313      * Gets the 'td' of the cell
56314      * 
56315      * @param {Integer} rowIndex row to select
56316      * @param {Integer} colIndex column to select
56317      * 
56318      * @return {Object} 
56319      */
56320     getCell : function(rowIndex, colIndex){
56321         var locked = this.cm.getLockedCount();
56322         var source;
56323         if(colIndex < locked){
56324             source = this.lockedBody.dom.firstChild;
56325         }else{
56326             source = this.mainBody.dom.firstChild;
56327             colIndex -= locked;
56328         }
56329         return source.rows[rowIndex].childNodes[colIndex];
56330     },
56331
56332     getCellText : function(rowIndex, colIndex){
56333         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56334     },
56335
56336     getCellBox : function(cell){
56337         var b = this.fly(cell).getBox();
56338         if(Roo.isOpera){ // opera fails to report the Y
56339             b.y = cell.offsetTop + this.mainBody.getY();
56340         }
56341         return b;
56342     },
56343
56344     getCellIndex : function(cell){
56345         var id = String(cell.className).match(this.cellRE);
56346         if(id){
56347             return parseInt(id[1], 10);
56348         }
56349         return 0;
56350     },
56351
56352     findHeaderIndex : function(n){
56353         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56354         return r ? this.getCellIndex(r) : false;
56355     },
56356
56357     findHeaderCell : function(n){
56358         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56359         return r ? r : false;
56360     },
56361
56362     findRowIndex : function(n){
56363         if(!n){
56364             return false;
56365         }
56366         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56367         return r ? r.rowIndex : false;
56368     },
56369
56370     findCellIndex : function(node){
56371         var stop = this.el.dom;
56372         while(node && node != stop){
56373             if(this.findRE.test(node.className)){
56374                 return this.getCellIndex(node);
56375             }
56376             node = node.parentNode;
56377         }
56378         return false;
56379     },
56380
56381     getColumnId : function(index){
56382         return this.cm.getColumnId(index);
56383     },
56384
56385     getSplitters : function()
56386     {
56387         if(this.splitterSelector){
56388            return Roo.DomQuery.select(this.splitterSelector);
56389         }else{
56390             return null;
56391       }
56392     },
56393
56394     getSplitter : function(index){
56395         return this.getSplitters()[index];
56396     },
56397
56398     onRowOver : function(e, t){
56399         var row;
56400         if((row = this.findRowIndex(t)) !== false){
56401             this.getRowComposite(row).addClass("x-grid-row-over");
56402         }
56403     },
56404
56405     onRowOut : function(e, t){
56406         var row;
56407         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56408             this.getRowComposite(row).removeClass("x-grid-row-over");
56409         }
56410     },
56411
56412     renderHeaders : function(){
56413         var cm = this.cm;
56414         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56415         var cb = [], lb = [], sb = [], lsb = [], p = {};
56416         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56417             p.cellId = "x-grid-hd-0-" + i;
56418             p.splitId = "x-grid-csplit-0-" + i;
56419             p.id = cm.getColumnId(i);
56420             p.value = cm.getColumnHeader(i) || "";
56421             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56422             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56423             if(!cm.isLocked(i)){
56424                 cb[cb.length] = ct.apply(p);
56425                 sb[sb.length] = st.apply(p);
56426             }else{
56427                 lb[lb.length] = ct.apply(p);
56428                 lsb[lsb.length] = st.apply(p);
56429             }
56430         }
56431         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56432                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56433     },
56434
56435     updateHeaders : function(){
56436         var html = this.renderHeaders();
56437         this.lockedHd.update(html[0]);
56438         this.mainHd.update(html[1]);
56439     },
56440
56441     /**
56442      * Focuses the specified row.
56443      * @param {Number} row The row index
56444      */
56445     focusRow : function(row)
56446     {
56447         //Roo.log('GridView.focusRow');
56448         var x = this.scroller.dom.scrollLeft;
56449         this.focusCell(row, 0, false);
56450         this.scroller.dom.scrollLeft = x;
56451     },
56452
56453     /**
56454      * Focuses the specified cell.
56455      * @param {Number} row The row index
56456      * @param {Number} col The column index
56457      * @param {Boolean} hscroll false to disable horizontal scrolling
56458      */
56459     focusCell : function(row, col, hscroll)
56460     {
56461         //Roo.log('GridView.focusCell');
56462         var el = this.ensureVisible(row, col, hscroll);
56463         this.focusEl.alignTo(el, "tl-tl");
56464         if(Roo.isGecko){
56465             this.focusEl.focus();
56466         }else{
56467             this.focusEl.focus.defer(1, this.focusEl);
56468         }
56469     },
56470
56471     /**
56472      * Scrolls the specified cell into view
56473      * @param {Number} row The row index
56474      * @param {Number} col The column index
56475      * @param {Boolean} hscroll false to disable horizontal scrolling
56476      */
56477     ensureVisible : function(row, col, hscroll)
56478     {
56479         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56480         //return null; //disable for testing.
56481         if(typeof row != "number"){
56482             row = row.rowIndex;
56483         }
56484         if(row < 0 && row >= this.ds.getCount()){
56485             return  null;
56486         }
56487         col = (col !== undefined ? col : 0);
56488         var cm = this.grid.colModel;
56489         while(cm.isHidden(col)){
56490             col++;
56491         }
56492
56493         var el = this.getCell(row, col);
56494         if(!el){
56495             return null;
56496         }
56497         var c = this.scroller.dom;
56498
56499         var ctop = parseInt(el.offsetTop, 10);
56500         var cleft = parseInt(el.offsetLeft, 10);
56501         var cbot = ctop + el.offsetHeight;
56502         var cright = cleft + el.offsetWidth;
56503         
56504         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56505         var stop = parseInt(c.scrollTop, 10);
56506         var sleft = parseInt(c.scrollLeft, 10);
56507         var sbot = stop + ch;
56508         var sright = sleft + c.clientWidth;
56509         /*
56510         Roo.log('GridView.ensureVisible:' +
56511                 ' ctop:' + ctop +
56512                 ' c.clientHeight:' + c.clientHeight +
56513                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56514                 ' stop:' + stop +
56515                 ' cbot:' + cbot +
56516                 ' sbot:' + sbot +
56517                 ' ch:' + ch  
56518                 );
56519         */
56520         if(ctop < stop){
56521              c.scrollTop = ctop;
56522             //Roo.log("set scrolltop to ctop DISABLE?");
56523         }else if(cbot > sbot){
56524             //Roo.log("set scrolltop to cbot-ch");
56525             c.scrollTop = cbot-ch;
56526         }
56527         
56528         if(hscroll !== false){
56529             if(cleft < sleft){
56530                 c.scrollLeft = cleft;
56531             }else if(cright > sright){
56532                 c.scrollLeft = cright-c.clientWidth;
56533             }
56534         }
56535          
56536         return el;
56537     },
56538
56539     updateColumns : function(){
56540         this.grid.stopEditing();
56541         var cm = this.grid.colModel, colIds = this.getColumnIds();
56542         //var totalWidth = cm.getTotalWidth();
56543         var pos = 0;
56544         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56545             //if(cm.isHidden(i)) continue;
56546             var w = cm.getColumnWidth(i);
56547             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56548             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56549         }
56550         this.updateSplitters();
56551     },
56552
56553     generateRules : function(cm){
56554         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56555         Roo.util.CSS.removeStyleSheet(rulesId);
56556         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56557             var cid = cm.getColumnId(i);
56558             var align = '';
56559             if(cm.config[i].align){
56560                 align = 'text-align:'+cm.config[i].align+';';
56561             }
56562             var hidden = '';
56563             if(cm.isHidden(i)){
56564                 hidden = 'display:none;';
56565             }
56566             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56567             ruleBuf.push(
56568                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56569                     this.hdSelector, cid, " {\n", align, width, "}\n",
56570                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56571                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56572         }
56573         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56574     },
56575
56576     updateSplitters : function(){
56577         var cm = this.cm, s = this.getSplitters();
56578         if(s){ // splitters not created yet
56579             var pos = 0, locked = true;
56580             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56581                 if(cm.isHidden(i)) {
56582                     continue;
56583                 }
56584                 var w = cm.getColumnWidth(i); // make sure it's a number
56585                 if(!cm.isLocked(i) && locked){
56586                     pos = 0;
56587                     locked = false;
56588                 }
56589                 pos += w;
56590                 s[i].style.left = (pos-this.splitOffset) + "px";
56591             }
56592         }
56593     },
56594
56595     handleHiddenChange : function(colModel, colIndex, hidden){
56596         if(hidden){
56597             this.hideColumn(colIndex);
56598         }else{
56599             this.unhideColumn(colIndex);
56600         }
56601     },
56602
56603     hideColumn : function(colIndex){
56604         var cid = this.getColumnId(colIndex);
56605         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56606         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56607         if(Roo.isSafari){
56608             this.updateHeaders();
56609         }
56610         this.updateSplitters();
56611         this.layout();
56612     },
56613
56614     unhideColumn : function(colIndex){
56615         var cid = this.getColumnId(colIndex);
56616         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56617         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56618
56619         if(Roo.isSafari){
56620             this.updateHeaders();
56621         }
56622         this.updateSplitters();
56623         this.layout();
56624     },
56625
56626     insertRows : function(dm, firstRow, lastRow, isUpdate){
56627         if(firstRow == 0 && lastRow == dm.getCount()-1){
56628             this.refresh();
56629         }else{
56630             if(!isUpdate){
56631                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56632             }
56633             var s = this.getScrollState();
56634             var markup = this.renderRows(firstRow, lastRow);
56635             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56636             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56637             this.restoreScroll(s);
56638             if(!isUpdate){
56639                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56640                 this.syncRowHeights(firstRow, lastRow);
56641                 this.stripeRows(firstRow);
56642                 this.layout();
56643             }
56644         }
56645     },
56646
56647     bufferRows : function(markup, target, index){
56648         var before = null, trows = target.rows, tbody = target.tBodies[0];
56649         if(index < trows.length){
56650             before = trows[index];
56651         }
56652         var b = document.createElement("div");
56653         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56654         var rows = b.firstChild.rows;
56655         for(var i = 0, len = rows.length; i < len; i++){
56656             if(before){
56657                 tbody.insertBefore(rows[0], before);
56658             }else{
56659                 tbody.appendChild(rows[0]);
56660             }
56661         }
56662         b.innerHTML = "";
56663         b = null;
56664     },
56665
56666     deleteRows : function(dm, firstRow, lastRow){
56667         if(dm.getRowCount()<1){
56668             this.fireEvent("beforerefresh", this);
56669             this.mainBody.update("");
56670             this.lockedBody.update("");
56671             this.fireEvent("refresh", this);
56672         }else{
56673             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56674             var bt = this.getBodyTable();
56675             var tbody = bt.firstChild;
56676             var rows = bt.rows;
56677             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56678                 tbody.removeChild(rows[firstRow]);
56679             }
56680             this.stripeRows(firstRow);
56681             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56682         }
56683     },
56684
56685     updateRows : function(dataSource, firstRow, lastRow){
56686         var s = this.getScrollState();
56687         this.refresh();
56688         this.restoreScroll(s);
56689     },
56690
56691     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56692         if(!noRefresh){
56693            this.refresh();
56694         }
56695         this.updateHeaderSortState();
56696     },
56697
56698     getScrollState : function(){
56699         
56700         var sb = this.scroller.dom;
56701         return {left: sb.scrollLeft, top: sb.scrollTop};
56702     },
56703
56704     stripeRows : function(startRow){
56705         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56706             return;
56707         }
56708         startRow = startRow || 0;
56709         var rows = this.getBodyTable().rows;
56710         var lrows = this.getLockedTable().rows;
56711         var cls = ' x-grid-row-alt ';
56712         for(var i = startRow, len = rows.length; i < len; i++){
56713             var row = rows[i], lrow = lrows[i];
56714             var isAlt = ((i+1) % 2 == 0);
56715             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56716             if(isAlt == hasAlt){
56717                 continue;
56718             }
56719             if(isAlt){
56720                 row.className += " x-grid-row-alt";
56721             }else{
56722                 row.className = row.className.replace("x-grid-row-alt", "");
56723             }
56724             if(lrow){
56725                 lrow.className = row.className;
56726             }
56727         }
56728     },
56729
56730     restoreScroll : function(state){
56731         //Roo.log('GridView.restoreScroll');
56732         var sb = this.scroller.dom;
56733         sb.scrollLeft = state.left;
56734         sb.scrollTop = state.top;
56735         this.syncScroll();
56736     },
56737
56738     syncScroll : function(){
56739         //Roo.log('GridView.syncScroll');
56740         var sb = this.scroller.dom;
56741         var sh = this.mainHd.dom;
56742         var bs = this.mainBody.dom;
56743         var lv = this.lockedBody.dom;
56744         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56745         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56746     },
56747
56748     handleScroll : function(e){
56749         this.syncScroll();
56750         var sb = this.scroller.dom;
56751         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56752         e.stopEvent();
56753     },
56754
56755     handleWheel : function(e){
56756         var d = e.getWheelDelta();
56757         this.scroller.dom.scrollTop -= d*22;
56758         // set this here to prevent jumpy scrolling on large tables
56759         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56760         e.stopEvent();
56761     },
56762
56763     renderRows : function(startRow, endRow){
56764         // pull in all the crap needed to render rows
56765         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56766         var colCount = cm.getColumnCount();
56767
56768         if(ds.getCount() < 1){
56769             return ["", ""];
56770         }
56771
56772         // build a map for all the columns
56773         var cs = [];
56774         for(var i = 0; i < colCount; i++){
56775             var name = cm.getDataIndex(i);
56776             cs[i] = {
56777                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56778                 renderer : cm.getRenderer(i),
56779                 id : cm.getColumnId(i),
56780                 locked : cm.isLocked(i),
56781                 has_editor : cm.isCellEditable(i)
56782             };
56783         }
56784
56785         startRow = startRow || 0;
56786         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56787
56788         // records to render
56789         var rs = ds.getRange(startRow, endRow);
56790
56791         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56792     },
56793
56794     // As much as I hate to duplicate code, this was branched because FireFox really hates
56795     // [].join("") on strings. The performance difference was substantial enough to
56796     // branch this function
56797     doRender : Roo.isGecko ?
56798             function(cs, rs, ds, startRow, colCount, stripe){
56799                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56800                 // buffers
56801                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56802                 
56803                 var hasListener = this.grid.hasListener('rowclass');
56804                 var rowcfg = {};
56805                 for(var j = 0, len = rs.length; j < len; j++){
56806                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56807                     for(var i = 0; i < colCount; i++){
56808                         c = cs[i];
56809                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56810                         p.id = c.id;
56811                         p.css = p.attr = "";
56812                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56813                         if(p.value == undefined || p.value === "") {
56814                             p.value = "&#160;";
56815                         }
56816                         if(c.has_editor){
56817                             p.css += ' x-grid-editable-cell';
56818                         }
56819                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56820                             p.css +=  ' x-grid-dirty-cell';
56821                         }
56822                         var markup = ct.apply(p);
56823                         if(!c.locked){
56824                             cb+= markup;
56825                         }else{
56826                             lcb+= markup;
56827                         }
56828                     }
56829                     var alt = [];
56830                     if(stripe && ((rowIndex+1) % 2 == 0)){
56831                         alt.push("x-grid-row-alt")
56832                     }
56833                     if(r.dirty){
56834                         alt.push(  " x-grid-dirty-row");
56835                     }
56836                     rp.cells = lcb;
56837                     if(this.getRowClass){
56838                         alt.push(this.getRowClass(r, rowIndex));
56839                     }
56840                     if (hasListener) {
56841                         rowcfg = {
56842                              
56843                             record: r,
56844                             rowIndex : rowIndex,
56845                             rowClass : ''
56846                         };
56847                         this.grid.fireEvent('rowclass', this, rowcfg);
56848                         alt.push(rowcfg.rowClass);
56849                     }
56850                     rp.alt = alt.join(" ");
56851                     lbuf+= rt.apply(rp);
56852                     rp.cells = cb;
56853                     buf+=  rt.apply(rp);
56854                 }
56855                 return [lbuf, buf];
56856             } :
56857             function(cs, rs, ds, startRow, colCount, stripe){
56858                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56859                 // buffers
56860                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56861                 var hasListener = this.grid.hasListener('rowclass');
56862  
56863                 var rowcfg = {};
56864                 for(var j = 0, len = rs.length; j < len; j++){
56865                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56866                     for(var i = 0; i < colCount; i++){
56867                         c = cs[i];
56868                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56869                         p.id = c.id;
56870                         p.css = p.attr = "";
56871                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56872                         if(p.value == undefined || p.value === "") {
56873                             p.value = "&#160;";
56874                         }
56875                         //Roo.log(c);
56876                          if(c.has_editor){
56877                             p.css += ' x-grid-editable-cell';
56878                         }
56879                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56880                             p.css += ' x-grid-dirty-cell' 
56881                         }
56882                         
56883                         var markup = ct.apply(p);
56884                         if(!c.locked){
56885                             cb[cb.length] = markup;
56886                         }else{
56887                             lcb[lcb.length] = markup;
56888                         }
56889                     }
56890                     var alt = [];
56891                     if(stripe && ((rowIndex+1) % 2 == 0)){
56892                         alt.push( "x-grid-row-alt");
56893                     }
56894                     if(r.dirty){
56895                         alt.push(" x-grid-dirty-row");
56896                     }
56897                     rp.cells = lcb;
56898                     if(this.getRowClass){
56899                         alt.push( this.getRowClass(r, rowIndex));
56900                     }
56901                     if (hasListener) {
56902                         rowcfg = {
56903                              
56904                             record: r,
56905                             rowIndex : rowIndex,
56906                             rowClass : ''
56907                         };
56908                         this.grid.fireEvent('rowclass', this, rowcfg);
56909                         alt.push(rowcfg.rowClass);
56910                     }
56911                     
56912                     rp.alt = alt.join(" ");
56913                     rp.cells = lcb.join("");
56914                     lbuf[lbuf.length] = rt.apply(rp);
56915                     rp.cells = cb.join("");
56916                     buf[buf.length] =  rt.apply(rp);
56917                 }
56918                 return [lbuf.join(""), buf.join("")];
56919             },
56920
56921     renderBody : function(){
56922         var markup = this.renderRows();
56923         var bt = this.templates.body;
56924         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56925     },
56926
56927     /**
56928      * Refreshes the grid
56929      * @param {Boolean} headersToo
56930      */
56931     refresh : function(headersToo){
56932         this.fireEvent("beforerefresh", this);
56933         this.grid.stopEditing();
56934         var result = this.renderBody();
56935         this.lockedBody.update(result[0]);
56936         this.mainBody.update(result[1]);
56937         if(headersToo === true){
56938             this.updateHeaders();
56939             this.updateColumns();
56940             this.updateSplitters();
56941             this.updateHeaderSortState();
56942         }
56943         this.syncRowHeights();
56944         this.layout();
56945         this.fireEvent("refresh", this);
56946     },
56947
56948     handleColumnMove : function(cm, oldIndex, newIndex){
56949         this.indexMap = null;
56950         var s = this.getScrollState();
56951         this.refresh(true);
56952         this.restoreScroll(s);
56953         this.afterMove(newIndex);
56954     },
56955
56956     afterMove : function(colIndex){
56957         if(this.enableMoveAnim && Roo.enableFx){
56958             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56959         }
56960         // if multisort - fix sortOrder, and reload..
56961         if (this.grid.dataSource.multiSort) {
56962             // the we can call sort again..
56963             var dm = this.grid.dataSource;
56964             var cm = this.grid.colModel;
56965             var so = [];
56966             for(var i = 0; i < cm.config.length; i++ ) {
56967                 
56968                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56969                     continue; // dont' bother, it's not in sort list or being set.
56970                 }
56971                 
56972                 so.push(cm.config[i].dataIndex);
56973             };
56974             dm.sortOrder = so;
56975             dm.load(dm.lastOptions);
56976             
56977             
56978         }
56979         
56980     },
56981
56982     updateCell : function(dm, rowIndex, dataIndex){
56983         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56984         if(typeof colIndex == "undefined"){ // not present in grid
56985             return;
56986         }
56987         var cm = this.grid.colModel;
56988         var cell = this.getCell(rowIndex, colIndex);
56989         var cellText = this.getCellText(rowIndex, colIndex);
56990
56991         var p = {
56992             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56993             id : cm.getColumnId(colIndex),
56994             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56995         };
56996         var renderer = cm.getRenderer(colIndex);
56997         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56998         if(typeof val == "undefined" || val === "") {
56999             val = "&#160;";
57000         }
57001         cellText.innerHTML = val;
57002         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
57003         this.syncRowHeights(rowIndex, rowIndex);
57004     },
57005
57006     calcColumnWidth : function(colIndex, maxRowsToMeasure){
57007         var maxWidth = 0;
57008         if(this.grid.autoSizeHeaders){
57009             var h = this.getHeaderCellMeasure(colIndex);
57010             maxWidth = Math.max(maxWidth, h.scrollWidth);
57011         }
57012         var tb, index;
57013         if(this.cm.isLocked(colIndex)){
57014             tb = this.getLockedTable();
57015             index = colIndex;
57016         }else{
57017             tb = this.getBodyTable();
57018             index = colIndex - this.cm.getLockedCount();
57019         }
57020         if(tb && tb.rows){
57021             var rows = tb.rows;
57022             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57023             for(var i = 0; i < stopIndex; i++){
57024                 var cell = rows[i].childNodes[index].firstChild;
57025                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57026             }
57027         }
57028         return maxWidth + /*margin for error in IE*/ 5;
57029     },
57030     /**
57031      * Autofit a column to its content.
57032      * @param {Number} colIndex
57033      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57034      */
57035      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57036          if(this.cm.isHidden(colIndex)){
57037              return; // can't calc a hidden column
57038          }
57039         if(forceMinSize){
57040             var cid = this.cm.getColumnId(colIndex);
57041             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57042            if(this.grid.autoSizeHeaders){
57043                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57044            }
57045         }
57046         var newWidth = this.calcColumnWidth(colIndex);
57047         this.cm.setColumnWidth(colIndex,
57048             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57049         if(!suppressEvent){
57050             this.grid.fireEvent("columnresize", colIndex, newWidth);
57051         }
57052     },
57053
57054     /**
57055      * Autofits all columns to their content and then expands to fit any extra space in the grid
57056      */
57057      autoSizeColumns : function(){
57058         var cm = this.grid.colModel;
57059         var colCount = cm.getColumnCount();
57060         for(var i = 0; i < colCount; i++){
57061             this.autoSizeColumn(i, true, true);
57062         }
57063         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57064             this.fitColumns();
57065         }else{
57066             this.updateColumns();
57067             this.layout();
57068         }
57069     },
57070
57071     /**
57072      * Autofits all columns to the grid's width proportionate with their current size
57073      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57074      */
57075     fitColumns : function(reserveScrollSpace){
57076         var cm = this.grid.colModel;
57077         var colCount = cm.getColumnCount();
57078         var cols = [];
57079         var width = 0;
57080         var i, w;
57081         for (i = 0; i < colCount; i++){
57082             if(!cm.isHidden(i) && !cm.isFixed(i)){
57083                 w = cm.getColumnWidth(i);
57084                 cols.push(i);
57085                 cols.push(w);
57086                 width += w;
57087             }
57088         }
57089         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57090         if(reserveScrollSpace){
57091             avail -= 17;
57092         }
57093         var frac = (avail - cm.getTotalWidth())/width;
57094         while (cols.length){
57095             w = cols.pop();
57096             i = cols.pop();
57097             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57098         }
57099         this.updateColumns();
57100         this.layout();
57101     },
57102
57103     onRowSelect : function(rowIndex){
57104         var row = this.getRowComposite(rowIndex);
57105         row.addClass("x-grid-row-selected");
57106     },
57107
57108     onRowDeselect : function(rowIndex){
57109         var row = this.getRowComposite(rowIndex);
57110         row.removeClass("x-grid-row-selected");
57111     },
57112
57113     onCellSelect : function(row, col){
57114         var cell = this.getCell(row, col);
57115         if(cell){
57116             Roo.fly(cell).addClass("x-grid-cell-selected");
57117         }
57118     },
57119
57120     onCellDeselect : function(row, col){
57121         var cell = this.getCell(row, col);
57122         if(cell){
57123             Roo.fly(cell).removeClass("x-grid-cell-selected");
57124         }
57125     },
57126
57127     updateHeaderSortState : function(){
57128         
57129         // sort state can be single { field: xxx, direction : yyy}
57130         // or   { xxx=>ASC , yyy : DESC ..... }
57131         
57132         var mstate = {};
57133         if (!this.ds.multiSort) { 
57134             var state = this.ds.getSortState();
57135             if(!state){
57136                 return;
57137             }
57138             mstate[state.field] = state.direction;
57139             // FIXME... - this is not used here.. but might be elsewhere..
57140             this.sortState = state;
57141             
57142         } else {
57143             mstate = this.ds.sortToggle;
57144         }
57145         //remove existing sort classes..
57146         
57147         var sc = this.sortClasses;
57148         var hds = this.el.select(this.headerSelector).removeClass(sc);
57149         
57150         for(var f in mstate) {
57151         
57152             var sortColumn = this.cm.findColumnIndex(f);
57153             
57154             if(sortColumn != -1){
57155                 var sortDir = mstate[f];        
57156                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57157             }
57158         }
57159         
57160          
57161         
57162     },
57163
57164
57165     handleHeaderClick : function(g, index,e){
57166         
57167         Roo.log("header click");
57168         
57169         if (Roo.isTouch) {
57170             // touch events on header are handled by context
57171             this.handleHdCtx(g,index,e);
57172             return;
57173         }
57174         
57175         
57176         if(this.headersDisabled){
57177             return;
57178         }
57179         var dm = g.dataSource, cm = g.colModel;
57180         if(!cm.isSortable(index)){
57181             return;
57182         }
57183         g.stopEditing();
57184         
57185         if (dm.multiSort) {
57186             // update the sortOrder
57187             var so = [];
57188             for(var i = 0; i < cm.config.length; i++ ) {
57189                 
57190                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57191                     continue; // dont' bother, it's not in sort list or being set.
57192                 }
57193                 
57194                 so.push(cm.config[i].dataIndex);
57195             };
57196             dm.sortOrder = so;
57197         }
57198         
57199         
57200         dm.sort(cm.getDataIndex(index));
57201     },
57202
57203
57204     destroy : function(){
57205         if(this.colMenu){
57206             this.colMenu.removeAll();
57207             Roo.menu.MenuMgr.unregister(this.colMenu);
57208             this.colMenu.getEl().remove();
57209             delete this.colMenu;
57210         }
57211         if(this.hmenu){
57212             this.hmenu.removeAll();
57213             Roo.menu.MenuMgr.unregister(this.hmenu);
57214             this.hmenu.getEl().remove();
57215             delete this.hmenu;
57216         }
57217         if(this.grid.enableColumnMove){
57218             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57219             if(dds){
57220                 for(var dd in dds){
57221                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57222                         var elid = dds[dd].dragElId;
57223                         dds[dd].unreg();
57224                         Roo.get(elid).remove();
57225                     } else if(dds[dd].config.isTarget){
57226                         dds[dd].proxyTop.remove();
57227                         dds[dd].proxyBottom.remove();
57228                         dds[dd].unreg();
57229                     }
57230                     if(Roo.dd.DDM.locationCache[dd]){
57231                         delete Roo.dd.DDM.locationCache[dd];
57232                     }
57233                 }
57234                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57235             }
57236         }
57237         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57238         this.bind(null, null);
57239         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57240     },
57241
57242     handleLockChange : function(){
57243         this.refresh(true);
57244     },
57245
57246     onDenyColumnLock : function(){
57247
57248     },
57249
57250     onDenyColumnHide : function(){
57251
57252     },
57253
57254     handleHdMenuClick : function(item){
57255         var index = this.hdCtxIndex;
57256         var cm = this.cm, ds = this.ds;
57257         switch(item.id){
57258             case "asc":
57259                 ds.sort(cm.getDataIndex(index), "ASC");
57260                 break;
57261             case "desc":
57262                 ds.sort(cm.getDataIndex(index), "DESC");
57263                 break;
57264             case "lock":
57265                 var lc = cm.getLockedCount();
57266                 if(cm.getColumnCount(true) <= lc+1){
57267                     this.onDenyColumnLock();
57268                     return;
57269                 }
57270                 if(lc != index){
57271                     cm.setLocked(index, true, true);
57272                     cm.moveColumn(index, lc);
57273                     this.grid.fireEvent("columnmove", index, lc);
57274                 }else{
57275                     cm.setLocked(index, true);
57276                 }
57277             break;
57278             case "unlock":
57279                 var lc = cm.getLockedCount();
57280                 if((lc-1) != index){
57281                     cm.setLocked(index, false, true);
57282                     cm.moveColumn(index, lc-1);
57283                     this.grid.fireEvent("columnmove", index, lc-1);
57284                 }else{
57285                     cm.setLocked(index, false);
57286                 }
57287             break;
57288             case 'wider': // used to expand cols on touch..
57289             case 'narrow':
57290                 var cw = cm.getColumnWidth(index);
57291                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57292                 cw = Math.max(0, cw);
57293                 cw = Math.min(cw,4000);
57294                 cm.setColumnWidth(index, cw);
57295                 break;
57296                 
57297             default:
57298                 index = cm.getIndexById(item.id.substr(4));
57299                 if(index != -1){
57300                     if(item.checked && cm.getColumnCount(true) <= 1){
57301                         this.onDenyColumnHide();
57302                         return false;
57303                     }
57304                     cm.setHidden(index, item.checked);
57305                 }
57306         }
57307         return true;
57308     },
57309
57310     beforeColMenuShow : function(){
57311         var cm = this.cm,  colCount = cm.getColumnCount();
57312         this.colMenu.removeAll();
57313         for(var i = 0; i < colCount; i++){
57314             this.colMenu.add(new Roo.menu.CheckItem({
57315                 id: "col-"+cm.getColumnId(i),
57316                 text: cm.getColumnHeader(i),
57317                 checked: !cm.isHidden(i),
57318                 hideOnClick:false
57319             }));
57320         }
57321     },
57322
57323     handleHdCtx : function(g, index, e){
57324         e.stopEvent();
57325         var hd = this.getHeaderCell(index);
57326         this.hdCtxIndex = index;
57327         var ms = this.hmenu.items, cm = this.cm;
57328         ms.get("asc").setDisabled(!cm.isSortable(index));
57329         ms.get("desc").setDisabled(!cm.isSortable(index));
57330         if(this.grid.enableColLock !== false){
57331             ms.get("lock").setDisabled(cm.isLocked(index));
57332             ms.get("unlock").setDisabled(!cm.isLocked(index));
57333         }
57334         this.hmenu.show(hd, "tl-bl");
57335     },
57336
57337     handleHdOver : function(e){
57338         var hd = this.findHeaderCell(e.getTarget());
57339         if(hd && !this.headersDisabled){
57340             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57341                this.fly(hd).addClass("x-grid-hd-over");
57342             }
57343         }
57344     },
57345
57346     handleHdOut : function(e){
57347         var hd = this.findHeaderCell(e.getTarget());
57348         if(hd){
57349             this.fly(hd).removeClass("x-grid-hd-over");
57350         }
57351     },
57352
57353     handleSplitDblClick : function(e, t){
57354         var i = this.getCellIndex(t);
57355         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57356             this.autoSizeColumn(i, true);
57357             this.layout();
57358         }
57359     },
57360
57361     render : function(){
57362
57363         var cm = this.cm;
57364         var colCount = cm.getColumnCount();
57365
57366         if(this.grid.monitorWindowResize === true){
57367             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57368         }
57369         var header = this.renderHeaders();
57370         var body = this.templates.body.apply({rows:""});
57371         var html = this.templates.master.apply({
57372             lockedBody: body,
57373             body: body,
57374             lockedHeader: header[0],
57375             header: header[1]
57376         });
57377
57378         //this.updateColumns();
57379
57380         this.grid.getGridEl().dom.innerHTML = html;
57381
57382         this.initElements();
57383         
57384         // a kludge to fix the random scolling effect in webkit
57385         this.el.on("scroll", function() {
57386             this.el.dom.scrollTop=0; // hopefully not recursive..
57387         },this);
57388
57389         this.scroller.on("scroll", this.handleScroll, this);
57390         this.lockedBody.on("mousewheel", this.handleWheel, this);
57391         this.mainBody.on("mousewheel", this.handleWheel, this);
57392
57393         this.mainHd.on("mouseover", this.handleHdOver, this);
57394         this.mainHd.on("mouseout", this.handleHdOut, this);
57395         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57396                 {delegate: "."+this.splitClass});
57397
57398         this.lockedHd.on("mouseover", this.handleHdOver, this);
57399         this.lockedHd.on("mouseout", this.handleHdOut, this);
57400         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57401                 {delegate: "."+this.splitClass});
57402
57403         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57404             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57405         }
57406
57407         this.updateSplitters();
57408
57409         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57410             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57411             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57412         }
57413
57414         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57415             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57416             this.hmenu.add(
57417                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57418                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57419             );
57420             if(this.grid.enableColLock !== false){
57421                 this.hmenu.add('-',
57422                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57423                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57424                 );
57425             }
57426             if (Roo.isTouch) {
57427                  this.hmenu.add('-',
57428                     {id:"wider", text: this.columnsWiderText},
57429                     {id:"narrow", text: this.columnsNarrowText }
57430                 );
57431                 
57432                  
57433             }
57434             
57435             if(this.grid.enableColumnHide !== false){
57436
57437                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57438                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57439                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57440
57441                 this.hmenu.add('-',
57442                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57443                 );
57444             }
57445             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57446
57447             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57448         }
57449
57450         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57451             this.dd = new Roo.grid.GridDragZone(this.grid, {
57452                 ddGroup : this.grid.ddGroup || 'GridDD'
57453             });
57454             
57455         }
57456
57457         /*
57458         for(var i = 0; i < colCount; i++){
57459             if(cm.isHidden(i)){
57460                 this.hideColumn(i);
57461             }
57462             if(cm.config[i].align){
57463                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57464                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57465             }
57466         }*/
57467         
57468         this.updateHeaderSortState();
57469
57470         this.beforeInitialResize();
57471         this.layout(true);
57472
57473         // two part rendering gives faster view to the user
57474         this.renderPhase2.defer(1, this);
57475     },
57476
57477     renderPhase2 : function(){
57478         // render the rows now
57479         this.refresh();
57480         if(this.grid.autoSizeColumns){
57481             this.autoSizeColumns();
57482         }
57483     },
57484
57485     beforeInitialResize : function(){
57486
57487     },
57488
57489     onColumnSplitterMoved : function(i, w){
57490         this.userResized = true;
57491         var cm = this.grid.colModel;
57492         cm.setColumnWidth(i, w, true);
57493         var cid = cm.getColumnId(i);
57494         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57495         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57496         this.updateSplitters();
57497         this.layout();
57498         this.grid.fireEvent("columnresize", i, w);
57499     },
57500
57501     syncRowHeights : function(startIndex, endIndex){
57502         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57503             startIndex = startIndex || 0;
57504             var mrows = this.getBodyTable().rows;
57505             var lrows = this.getLockedTable().rows;
57506             var len = mrows.length-1;
57507             endIndex = Math.min(endIndex || len, len);
57508             for(var i = startIndex; i <= endIndex; i++){
57509                 var m = mrows[i], l = lrows[i];
57510                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57511                 m.style.height = l.style.height = h + "px";
57512             }
57513         }
57514     },
57515
57516     layout : function(initialRender, is2ndPass){
57517         var g = this.grid;
57518         var auto = g.autoHeight;
57519         var scrollOffset = 16;
57520         var c = g.getGridEl(), cm = this.cm,
57521                 expandCol = g.autoExpandColumn,
57522                 gv = this;
57523         //c.beginMeasure();
57524
57525         if(!c.dom.offsetWidth){ // display:none?
57526             if(initialRender){
57527                 this.lockedWrap.show();
57528                 this.mainWrap.show();
57529             }
57530             return;
57531         }
57532
57533         var hasLock = this.cm.isLocked(0);
57534
57535         var tbh = this.headerPanel.getHeight();
57536         var bbh = this.footerPanel.getHeight();
57537
57538         if(auto){
57539             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57540             var newHeight = ch + c.getBorderWidth("tb");
57541             if(g.maxHeight){
57542                 newHeight = Math.min(g.maxHeight, newHeight);
57543             }
57544             c.setHeight(newHeight);
57545         }
57546
57547         if(g.autoWidth){
57548             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57549         }
57550
57551         var s = this.scroller;
57552
57553         var csize = c.getSize(true);
57554
57555         this.el.setSize(csize.width, csize.height);
57556
57557         this.headerPanel.setWidth(csize.width);
57558         this.footerPanel.setWidth(csize.width);
57559
57560         var hdHeight = this.mainHd.getHeight();
57561         var vw = csize.width;
57562         var vh = csize.height - (tbh + bbh);
57563
57564         s.setSize(vw, vh);
57565
57566         var bt = this.getBodyTable();
57567         
57568         if(cm.getLockedCount() == cm.config.length){
57569             bt = this.getLockedTable();
57570         }
57571         
57572         var ltWidth = hasLock ?
57573                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57574
57575         var scrollHeight = bt.offsetHeight;
57576         var scrollWidth = ltWidth + bt.offsetWidth;
57577         var vscroll = false, hscroll = false;
57578
57579         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57580
57581         var lw = this.lockedWrap, mw = this.mainWrap;
57582         var lb = this.lockedBody, mb = this.mainBody;
57583
57584         setTimeout(function(){
57585             var t = s.dom.offsetTop;
57586             var w = s.dom.clientWidth,
57587                 h = s.dom.clientHeight;
57588
57589             lw.setTop(t);
57590             lw.setSize(ltWidth, h);
57591
57592             mw.setLeftTop(ltWidth, t);
57593             mw.setSize(w-ltWidth, h);
57594
57595             lb.setHeight(h-hdHeight);
57596             mb.setHeight(h-hdHeight);
57597
57598             if(is2ndPass !== true && !gv.userResized && expandCol){
57599                 // high speed resize without full column calculation
57600                 
57601                 var ci = cm.getIndexById(expandCol);
57602                 if (ci < 0) {
57603                     ci = cm.findColumnIndex(expandCol);
57604                 }
57605                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57606                 var expandId = cm.getColumnId(ci);
57607                 var  tw = cm.getTotalWidth(false);
57608                 var currentWidth = cm.getColumnWidth(ci);
57609                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57610                 if(currentWidth != cw){
57611                     cm.setColumnWidth(ci, cw, true);
57612                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57613                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57614                     gv.updateSplitters();
57615                     gv.layout(false, true);
57616                 }
57617             }
57618
57619             if(initialRender){
57620                 lw.show();
57621                 mw.show();
57622             }
57623             //c.endMeasure();
57624         }, 10);
57625     },
57626
57627     onWindowResize : function(){
57628         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57629             return;
57630         }
57631         this.layout();
57632     },
57633
57634     appendFooter : function(parentEl){
57635         return null;
57636     },
57637
57638     sortAscText : "Sort Ascending",
57639     sortDescText : "Sort Descending",
57640     lockText : "Lock Column",
57641     unlockText : "Unlock Column",
57642     columnsText : "Columns",
57643  
57644     columnsWiderText : "Wider",
57645     columnsNarrowText : "Thinner"
57646 });
57647
57648
57649 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57650     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57651     this.proxy.el.addClass('x-grid3-col-dd');
57652 };
57653
57654 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57655     handleMouseDown : function(e){
57656
57657     },
57658
57659     callHandleMouseDown : function(e){
57660         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57661     }
57662 });
57663 /*
57664  * Based on:
57665  * Ext JS Library 1.1.1
57666  * Copyright(c) 2006-2007, Ext JS, LLC.
57667  *
57668  * Originally Released Under LGPL - original licence link has changed is not relivant.
57669  *
57670  * Fork - LGPL
57671  * <script type="text/javascript">
57672  */
57673  
57674 // private
57675 // This is a support class used internally by the Grid components
57676 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57677     this.grid = grid;
57678     this.view = grid.getView();
57679     this.proxy = this.view.resizeProxy;
57680     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57681         "gridSplitters" + this.grid.getGridEl().id, {
57682         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57683     });
57684     this.setHandleElId(Roo.id(hd));
57685     this.setOuterHandleElId(Roo.id(hd2));
57686     this.scroll = false;
57687 };
57688 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57689     fly: Roo.Element.fly,
57690
57691     b4StartDrag : function(x, y){
57692         this.view.headersDisabled = true;
57693         this.proxy.setHeight(this.view.mainWrap.getHeight());
57694         var w = this.cm.getColumnWidth(this.cellIndex);
57695         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57696         this.resetConstraints();
57697         this.setXConstraint(minw, 1000);
57698         this.setYConstraint(0, 0);
57699         this.minX = x - minw;
57700         this.maxX = x + 1000;
57701         this.startPos = x;
57702         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57703     },
57704
57705
57706     handleMouseDown : function(e){
57707         ev = Roo.EventObject.setEvent(e);
57708         var t = this.fly(ev.getTarget());
57709         if(t.hasClass("x-grid-split")){
57710             this.cellIndex = this.view.getCellIndex(t.dom);
57711             this.split = t.dom;
57712             this.cm = this.grid.colModel;
57713             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57714                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57715             }
57716         }
57717     },
57718
57719     endDrag : function(e){
57720         this.view.headersDisabled = false;
57721         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57722         var diff = endX - this.startPos;
57723         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57724     },
57725
57726     autoOffset : function(){
57727         this.setDelta(0,0);
57728     }
57729 });/*
57730  * Based on:
57731  * Ext JS Library 1.1.1
57732  * Copyright(c) 2006-2007, Ext JS, LLC.
57733  *
57734  * Originally Released Under LGPL - original licence link has changed is not relivant.
57735  *
57736  * Fork - LGPL
57737  * <script type="text/javascript">
57738  */
57739  
57740 // private
57741 // This is a support class used internally by the Grid components
57742 Roo.grid.GridDragZone = function(grid, config){
57743     this.view = grid.getView();
57744     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57745     if(this.view.lockedBody){
57746         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57747         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57748     }
57749     this.scroll = false;
57750     this.grid = grid;
57751     this.ddel = document.createElement('div');
57752     this.ddel.className = 'x-grid-dd-wrap';
57753 };
57754
57755 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57756     ddGroup : "GridDD",
57757
57758     getDragData : function(e){
57759         var t = Roo.lib.Event.getTarget(e);
57760         var rowIndex = this.view.findRowIndex(t);
57761         var sm = this.grid.selModel;
57762             
57763         //Roo.log(rowIndex);
57764         
57765         if (sm.getSelectedCell) {
57766             // cell selection..
57767             if (!sm.getSelectedCell()) {
57768                 return false;
57769             }
57770             if (rowIndex != sm.getSelectedCell()[0]) {
57771                 return false;
57772             }
57773         
57774         }
57775         
57776         if(rowIndex !== false){
57777             
57778             // if editorgrid.. 
57779             
57780             
57781             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57782                
57783             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57784               //  
57785             //}
57786             if (e.hasModifier()){
57787                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57788             }
57789             
57790             Roo.log("getDragData");
57791             
57792             return {
57793                 grid: this.grid,
57794                 ddel: this.ddel,
57795                 rowIndex: rowIndex,
57796                 selections:sm.getSelections ? sm.getSelections() : (
57797                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57798                 )
57799             };
57800         }
57801         return false;
57802     },
57803
57804     onInitDrag : function(e){
57805         var data = this.dragData;
57806         this.ddel.innerHTML = this.grid.getDragDropText();
57807         this.proxy.update(this.ddel);
57808         // fire start drag?
57809     },
57810
57811     afterRepair : function(){
57812         this.dragging = false;
57813     },
57814
57815     getRepairXY : function(e, data){
57816         return false;
57817     },
57818
57819     onEndDrag : function(data, e){
57820         // fire end drag?
57821     },
57822
57823     onValidDrop : function(dd, e, id){
57824         // fire drag drop?
57825         this.hideProxy();
57826     },
57827
57828     beforeInvalidDrop : function(e, id){
57829
57830     }
57831 });/*
57832  * Based on:
57833  * Ext JS Library 1.1.1
57834  * Copyright(c) 2006-2007, Ext JS, LLC.
57835  *
57836  * Originally Released Under LGPL - original licence link has changed is not relivant.
57837  *
57838  * Fork - LGPL
57839  * <script type="text/javascript">
57840  */
57841  
57842
57843 /**
57844  * @class Roo.grid.ColumnModel
57845  * @extends Roo.util.Observable
57846  * This is the default implementation of a ColumnModel used by the Grid. It defines
57847  * the columns in the grid.
57848  * <br>Usage:<br>
57849  <pre><code>
57850  var colModel = new Roo.grid.ColumnModel([
57851         {header: "Ticker", width: 60, sortable: true, locked: true},
57852         {header: "Company Name", width: 150, sortable: true},
57853         {header: "Market Cap.", width: 100, sortable: true},
57854         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57855         {header: "Employees", width: 100, sortable: true, resizable: false}
57856  ]);
57857  </code></pre>
57858  * <p>
57859  
57860  * The config options listed for this class are options which may appear in each
57861  * individual column definition.
57862  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57863  * @constructor
57864  * @param {Object} config An Array of column config objects. See this class's
57865  * config objects for details.
57866 */
57867 Roo.grid.ColumnModel = function(config){
57868         /**
57869      * The config passed into the constructor
57870      */
57871     this.config = config;
57872     this.lookup = {};
57873
57874     // if no id, create one
57875     // if the column does not have a dataIndex mapping,
57876     // map it to the order it is in the config
57877     for(var i = 0, len = config.length; i < len; i++){
57878         var c = config[i];
57879         if(typeof c.dataIndex == "undefined"){
57880             c.dataIndex = i;
57881         }
57882         if(typeof c.renderer == "string"){
57883             c.renderer = Roo.util.Format[c.renderer];
57884         }
57885         if(typeof c.id == "undefined"){
57886             c.id = Roo.id();
57887         }
57888         if(c.editor && c.editor.xtype){
57889             c.editor  = Roo.factory(c.editor, Roo.grid);
57890         }
57891         if(c.editor && c.editor.isFormField){
57892             c.editor = new Roo.grid.GridEditor(c.editor);
57893         }
57894         this.lookup[c.id] = c;
57895     }
57896
57897     /**
57898      * The width of columns which have no width specified (defaults to 100)
57899      * @type Number
57900      */
57901     this.defaultWidth = 100;
57902
57903     /**
57904      * Default sortable of columns which have no sortable specified (defaults to false)
57905      * @type Boolean
57906      */
57907     this.defaultSortable = false;
57908
57909     this.addEvents({
57910         /**
57911              * @event widthchange
57912              * Fires when the width of a column changes.
57913              * @param {ColumnModel} this
57914              * @param {Number} columnIndex The column index
57915              * @param {Number} newWidth The new width
57916              */
57917             "widthchange": true,
57918         /**
57919              * @event headerchange
57920              * Fires when the text of a header changes.
57921              * @param {ColumnModel} this
57922              * @param {Number} columnIndex The column index
57923              * @param {Number} newText The new header text
57924              */
57925             "headerchange": true,
57926         /**
57927              * @event hiddenchange
57928              * Fires when a column is hidden or "unhidden".
57929              * @param {ColumnModel} this
57930              * @param {Number} columnIndex The column index
57931              * @param {Boolean} hidden true if hidden, false otherwise
57932              */
57933             "hiddenchange": true,
57934             /**
57935          * @event columnmoved
57936          * Fires when a column is moved.
57937          * @param {ColumnModel} this
57938          * @param {Number} oldIndex
57939          * @param {Number} newIndex
57940          */
57941         "columnmoved" : true,
57942         /**
57943          * @event columlockchange
57944          * Fires when a column's locked state is changed
57945          * @param {ColumnModel} this
57946          * @param {Number} colIndex
57947          * @param {Boolean} locked true if locked
57948          */
57949         "columnlockchange" : true
57950     });
57951     Roo.grid.ColumnModel.superclass.constructor.call(this);
57952 };
57953 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57954     /**
57955      * @cfg {String} header The header text to display in the Grid view.
57956      */
57957     /**
57958      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57959      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57960      * specified, the column's index is used as an index into the Record's data Array.
57961      */
57962     /**
57963      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57964      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57965      */
57966     /**
57967      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57968      * Defaults to the value of the {@link #defaultSortable} property.
57969      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57970      */
57971     /**
57972      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57973      */
57974     /**
57975      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57976      */
57977     /**
57978      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57979      */
57980     /**
57981      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57982      */
57983     /**
57984      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57985      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57986      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57987      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57988      */
57989        /**
57990      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57991      */
57992     /**
57993      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57994      */
57995     /**
57996      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57997      */
57998     /**
57999      * @cfg {String} cursor (Optional)
58000      */
58001     /**
58002      * @cfg {String} tooltip (Optional)
58003      */
58004     /**
58005      * @cfg {Number} xs (Optional)
58006      */
58007     /**
58008      * @cfg {Number} sm (Optional)
58009      */
58010     /**
58011      * @cfg {Number} md (Optional)
58012      */
58013     /**
58014      * @cfg {Number} lg (Optional)
58015      */
58016     /**
58017      * Returns the id of the column at the specified index.
58018      * @param {Number} index The column index
58019      * @return {String} the id
58020      */
58021     getColumnId : function(index){
58022         return this.config[index].id;
58023     },
58024
58025     /**
58026      * Returns the column for a specified id.
58027      * @param {String} id The column id
58028      * @return {Object} the column
58029      */
58030     getColumnById : function(id){
58031         return this.lookup[id];
58032     },
58033
58034     
58035     /**
58036      * Returns the column for a specified dataIndex.
58037      * @param {String} dataIndex The column dataIndex
58038      * @return {Object|Boolean} the column or false if not found
58039      */
58040     getColumnByDataIndex: function(dataIndex){
58041         var index = this.findColumnIndex(dataIndex);
58042         return index > -1 ? this.config[index] : false;
58043     },
58044     
58045     /**
58046      * Returns the index for a specified column id.
58047      * @param {String} id The column id
58048      * @return {Number} the index, or -1 if not found
58049      */
58050     getIndexById : function(id){
58051         for(var i = 0, len = this.config.length; i < len; i++){
58052             if(this.config[i].id == id){
58053                 return i;
58054             }
58055         }
58056         return -1;
58057     },
58058     
58059     /**
58060      * Returns the index for a specified column dataIndex.
58061      * @param {String} dataIndex The column dataIndex
58062      * @return {Number} the index, or -1 if not found
58063      */
58064     
58065     findColumnIndex : function(dataIndex){
58066         for(var i = 0, len = this.config.length; i < len; i++){
58067             if(this.config[i].dataIndex == dataIndex){
58068                 return i;
58069             }
58070         }
58071         return -1;
58072     },
58073     
58074     
58075     moveColumn : function(oldIndex, newIndex){
58076         var c = this.config[oldIndex];
58077         this.config.splice(oldIndex, 1);
58078         this.config.splice(newIndex, 0, c);
58079         this.dataMap = null;
58080         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58081     },
58082
58083     isLocked : function(colIndex){
58084         return this.config[colIndex].locked === true;
58085     },
58086
58087     setLocked : function(colIndex, value, suppressEvent){
58088         if(this.isLocked(colIndex) == value){
58089             return;
58090         }
58091         this.config[colIndex].locked = value;
58092         if(!suppressEvent){
58093             this.fireEvent("columnlockchange", this, colIndex, value);
58094         }
58095     },
58096
58097     getTotalLockedWidth : function(){
58098         var totalWidth = 0;
58099         for(var i = 0; i < this.config.length; i++){
58100             if(this.isLocked(i) && !this.isHidden(i)){
58101                 this.totalWidth += this.getColumnWidth(i);
58102             }
58103         }
58104         return totalWidth;
58105     },
58106
58107     getLockedCount : function(){
58108         for(var i = 0, len = this.config.length; i < len; i++){
58109             if(!this.isLocked(i)){
58110                 return i;
58111             }
58112         }
58113         
58114         return this.config.length;
58115     },
58116
58117     /**
58118      * Returns the number of columns.
58119      * @return {Number}
58120      */
58121     getColumnCount : function(visibleOnly){
58122         if(visibleOnly === true){
58123             var c = 0;
58124             for(var i = 0, len = this.config.length; i < len; i++){
58125                 if(!this.isHidden(i)){
58126                     c++;
58127                 }
58128             }
58129             return c;
58130         }
58131         return this.config.length;
58132     },
58133
58134     /**
58135      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58136      * @param {Function} fn
58137      * @param {Object} scope (optional)
58138      * @return {Array} result
58139      */
58140     getColumnsBy : function(fn, scope){
58141         var r = [];
58142         for(var i = 0, len = this.config.length; i < len; i++){
58143             var c = this.config[i];
58144             if(fn.call(scope||this, c, i) === true){
58145                 r[r.length] = c;
58146             }
58147         }
58148         return r;
58149     },
58150
58151     /**
58152      * Returns true if the specified column is sortable.
58153      * @param {Number} col The column index
58154      * @return {Boolean}
58155      */
58156     isSortable : function(col){
58157         if(typeof this.config[col].sortable == "undefined"){
58158             return this.defaultSortable;
58159         }
58160         return this.config[col].sortable;
58161     },
58162
58163     /**
58164      * Returns the rendering (formatting) function defined for the column.
58165      * @param {Number} col The column index.
58166      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58167      */
58168     getRenderer : function(col){
58169         if(!this.config[col].renderer){
58170             return Roo.grid.ColumnModel.defaultRenderer;
58171         }
58172         return this.config[col].renderer;
58173     },
58174
58175     /**
58176      * Sets the rendering (formatting) function for a column.
58177      * @param {Number} col The column index
58178      * @param {Function} fn The function to use to process the cell's raw data
58179      * to return HTML markup for the grid view. The render function is called with
58180      * the following parameters:<ul>
58181      * <li>Data value.</li>
58182      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58183      * <li>css A CSS style string to apply to the table cell.</li>
58184      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58185      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58186      * <li>Row index</li>
58187      * <li>Column index</li>
58188      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58189      */
58190     setRenderer : function(col, fn){
58191         this.config[col].renderer = fn;
58192     },
58193
58194     /**
58195      * Returns the width for the specified column.
58196      * @param {Number} col The column index
58197      * @return {Number}
58198      */
58199     getColumnWidth : function(col){
58200         return this.config[col].width * 1 || this.defaultWidth;
58201     },
58202
58203     /**
58204      * Sets the width for a column.
58205      * @param {Number} col The column index
58206      * @param {Number} width The new width
58207      */
58208     setColumnWidth : function(col, width, suppressEvent){
58209         this.config[col].width = width;
58210         this.totalWidth = null;
58211         if(!suppressEvent){
58212              this.fireEvent("widthchange", this, col, width);
58213         }
58214     },
58215
58216     /**
58217      * Returns the total width of all columns.
58218      * @param {Boolean} includeHidden True to include hidden column widths
58219      * @return {Number}
58220      */
58221     getTotalWidth : function(includeHidden){
58222         if(!this.totalWidth){
58223             this.totalWidth = 0;
58224             for(var i = 0, len = this.config.length; i < len; i++){
58225                 if(includeHidden || !this.isHidden(i)){
58226                     this.totalWidth += this.getColumnWidth(i);
58227                 }
58228             }
58229         }
58230         return this.totalWidth;
58231     },
58232
58233     /**
58234      * Returns the header for the specified column.
58235      * @param {Number} col The column index
58236      * @return {String}
58237      */
58238     getColumnHeader : function(col){
58239         return this.config[col].header;
58240     },
58241
58242     /**
58243      * Sets the header for a column.
58244      * @param {Number} col The column index
58245      * @param {String} header The new header
58246      */
58247     setColumnHeader : function(col, header){
58248         this.config[col].header = header;
58249         this.fireEvent("headerchange", this, col, header);
58250     },
58251
58252     /**
58253      * Returns the tooltip for the specified column.
58254      * @param {Number} col The column index
58255      * @return {String}
58256      */
58257     getColumnTooltip : function(col){
58258             return this.config[col].tooltip;
58259     },
58260     /**
58261      * Sets the tooltip for a column.
58262      * @param {Number} col The column index
58263      * @param {String} tooltip The new tooltip
58264      */
58265     setColumnTooltip : function(col, tooltip){
58266             this.config[col].tooltip = tooltip;
58267     },
58268
58269     /**
58270      * Returns the dataIndex for the specified column.
58271      * @param {Number} col The column index
58272      * @return {Number}
58273      */
58274     getDataIndex : function(col){
58275         return this.config[col].dataIndex;
58276     },
58277
58278     /**
58279      * Sets the dataIndex for a column.
58280      * @param {Number} col The column index
58281      * @param {Number} dataIndex The new dataIndex
58282      */
58283     setDataIndex : function(col, dataIndex){
58284         this.config[col].dataIndex = dataIndex;
58285     },
58286
58287     
58288     
58289     /**
58290      * Returns true if the cell is editable.
58291      * @param {Number} colIndex The column index
58292      * @param {Number} rowIndex The row index - this is nto actually used..?
58293      * @return {Boolean}
58294      */
58295     isCellEditable : function(colIndex, rowIndex){
58296         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58297     },
58298
58299     /**
58300      * Returns the editor defined for the cell/column.
58301      * return false or null to disable editing.
58302      * @param {Number} colIndex The column index
58303      * @param {Number} rowIndex The row index
58304      * @return {Object}
58305      */
58306     getCellEditor : function(colIndex, rowIndex){
58307         return this.config[colIndex].editor;
58308     },
58309
58310     /**
58311      * Sets if a column is editable.
58312      * @param {Number} col The column index
58313      * @param {Boolean} editable True if the column is editable
58314      */
58315     setEditable : function(col, editable){
58316         this.config[col].editable = editable;
58317     },
58318
58319
58320     /**
58321      * Returns true if the column is hidden.
58322      * @param {Number} colIndex The column index
58323      * @return {Boolean}
58324      */
58325     isHidden : function(colIndex){
58326         return this.config[colIndex].hidden;
58327     },
58328
58329
58330     /**
58331      * Returns true if the column width cannot be changed
58332      */
58333     isFixed : function(colIndex){
58334         return this.config[colIndex].fixed;
58335     },
58336
58337     /**
58338      * Returns true if the column can be resized
58339      * @return {Boolean}
58340      */
58341     isResizable : function(colIndex){
58342         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58343     },
58344     /**
58345      * Sets if a column is hidden.
58346      * @param {Number} colIndex The column index
58347      * @param {Boolean} hidden True if the column is hidden
58348      */
58349     setHidden : function(colIndex, hidden){
58350         this.config[colIndex].hidden = hidden;
58351         this.totalWidth = null;
58352         this.fireEvent("hiddenchange", this, colIndex, hidden);
58353     },
58354
58355     /**
58356      * Sets the editor for a column.
58357      * @param {Number} col The column index
58358      * @param {Object} editor The editor object
58359      */
58360     setEditor : function(col, editor){
58361         this.config[col].editor = editor;
58362     }
58363 });
58364
58365 Roo.grid.ColumnModel.defaultRenderer = function(value)
58366 {
58367     if(typeof value == "object") {
58368         return value;
58369     }
58370         if(typeof value == "string" && value.length < 1){
58371             return "&#160;";
58372         }
58373     
58374         return String.format("{0}", value);
58375 };
58376
58377 // Alias for backwards compatibility
58378 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58379 /*
58380  * Based on:
58381  * Ext JS Library 1.1.1
58382  * Copyright(c) 2006-2007, Ext JS, LLC.
58383  *
58384  * Originally Released Under LGPL - original licence link has changed is not relivant.
58385  *
58386  * Fork - LGPL
58387  * <script type="text/javascript">
58388  */
58389
58390 /**
58391  * @class Roo.grid.AbstractSelectionModel
58392  * @extends Roo.util.Observable
58393  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58394  * implemented by descendant classes.  This class should not be directly instantiated.
58395  * @constructor
58396  */
58397 Roo.grid.AbstractSelectionModel = function(){
58398     this.locked = false;
58399     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58400 };
58401
58402 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58403     /** @ignore Called by the grid automatically. Do not call directly. */
58404     init : function(grid){
58405         this.grid = grid;
58406         this.initEvents();
58407     },
58408
58409     /**
58410      * Locks the selections.
58411      */
58412     lock : function(){
58413         this.locked = true;
58414     },
58415
58416     /**
58417      * Unlocks the selections.
58418      */
58419     unlock : function(){
58420         this.locked = false;
58421     },
58422
58423     /**
58424      * Returns true if the selections are locked.
58425      * @return {Boolean}
58426      */
58427     isLocked : function(){
58428         return this.locked;
58429     }
58430 });/*
58431  * Based on:
58432  * Ext JS Library 1.1.1
58433  * Copyright(c) 2006-2007, Ext JS, LLC.
58434  *
58435  * Originally Released Under LGPL - original licence link has changed is not relivant.
58436  *
58437  * Fork - LGPL
58438  * <script type="text/javascript">
58439  */
58440 /**
58441  * @extends Roo.grid.AbstractSelectionModel
58442  * @class Roo.grid.RowSelectionModel
58443  * The default SelectionModel used by {@link Roo.grid.Grid}.
58444  * It supports multiple selections and keyboard selection/navigation. 
58445  * @constructor
58446  * @param {Object} config
58447  */
58448 Roo.grid.RowSelectionModel = function(config){
58449     Roo.apply(this, config);
58450     this.selections = new Roo.util.MixedCollection(false, function(o){
58451         return o.id;
58452     });
58453
58454     this.last = false;
58455     this.lastActive = false;
58456
58457     this.addEvents({
58458         /**
58459              * @event selectionchange
58460              * Fires when the selection changes
58461              * @param {SelectionModel} this
58462              */
58463             "selectionchange" : true,
58464         /**
58465              * @event afterselectionchange
58466              * Fires after the selection changes (eg. by key press or clicking)
58467              * @param {SelectionModel} this
58468              */
58469             "afterselectionchange" : true,
58470         /**
58471              * @event beforerowselect
58472              * Fires when a row is selected being selected, return false to cancel.
58473              * @param {SelectionModel} this
58474              * @param {Number} rowIndex The selected index
58475              * @param {Boolean} keepExisting False if other selections will be cleared
58476              */
58477             "beforerowselect" : true,
58478         /**
58479              * @event rowselect
58480              * Fires when a row is selected.
58481              * @param {SelectionModel} this
58482              * @param {Number} rowIndex The selected index
58483              * @param {Roo.data.Record} r The record
58484              */
58485             "rowselect" : true,
58486         /**
58487              * @event rowdeselect
58488              * Fires when a row is deselected.
58489              * @param {SelectionModel} this
58490              * @param {Number} rowIndex The selected index
58491              */
58492         "rowdeselect" : true
58493     });
58494     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58495     this.locked = false;
58496 };
58497
58498 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58499     /**
58500      * @cfg {Boolean} singleSelect
58501      * True to allow selection of only one row at a time (defaults to false)
58502      */
58503     singleSelect : false,
58504
58505     // private
58506     initEvents : function(){
58507
58508         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58509             this.grid.on("mousedown", this.handleMouseDown, this);
58510         }else{ // allow click to work like normal
58511             this.grid.on("rowclick", this.handleDragableRowClick, this);
58512         }
58513
58514         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58515             "up" : function(e){
58516                 if(!e.shiftKey){
58517                     this.selectPrevious(e.shiftKey);
58518                 }else if(this.last !== false && this.lastActive !== false){
58519                     var last = this.last;
58520                     this.selectRange(this.last,  this.lastActive-1);
58521                     this.grid.getView().focusRow(this.lastActive);
58522                     if(last !== false){
58523                         this.last = last;
58524                     }
58525                 }else{
58526                     this.selectFirstRow();
58527                 }
58528                 this.fireEvent("afterselectionchange", this);
58529             },
58530             "down" : function(e){
58531                 if(!e.shiftKey){
58532                     this.selectNext(e.shiftKey);
58533                 }else if(this.last !== false && this.lastActive !== false){
58534                     var last = this.last;
58535                     this.selectRange(this.last,  this.lastActive+1);
58536                     this.grid.getView().focusRow(this.lastActive);
58537                     if(last !== false){
58538                         this.last = last;
58539                     }
58540                 }else{
58541                     this.selectFirstRow();
58542                 }
58543                 this.fireEvent("afterselectionchange", this);
58544             },
58545             scope: this
58546         });
58547
58548         var view = this.grid.view;
58549         view.on("refresh", this.onRefresh, this);
58550         view.on("rowupdated", this.onRowUpdated, this);
58551         view.on("rowremoved", this.onRemove, this);
58552     },
58553
58554     // private
58555     onRefresh : function(){
58556         var ds = this.grid.dataSource, i, v = this.grid.view;
58557         var s = this.selections;
58558         s.each(function(r){
58559             if((i = ds.indexOfId(r.id)) != -1){
58560                 v.onRowSelect(i);
58561                 s.add(ds.getAt(i)); // updating the selection relate data
58562             }else{
58563                 s.remove(r);
58564             }
58565         });
58566     },
58567
58568     // private
58569     onRemove : function(v, index, r){
58570         this.selections.remove(r);
58571     },
58572
58573     // private
58574     onRowUpdated : function(v, index, r){
58575         if(this.isSelected(r)){
58576             v.onRowSelect(index);
58577         }
58578     },
58579
58580     /**
58581      * Select records.
58582      * @param {Array} records The records to select
58583      * @param {Boolean} keepExisting (optional) True to keep existing selections
58584      */
58585     selectRecords : function(records, keepExisting){
58586         if(!keepExisting){
58587             this.clearSelections();
58588         }
58589         var ds = this.grid.dataSource;
58590         for(var i = 0, len = records.length; i < len; i++){
58591             this.selectRow(ds.indexOf(records[i]), true);
58592         }
58593     },
58594
58595     /**
58596      * Gets the number of selected rows.
58597      * @return {Number}
58598      */
58599     getCount : function(){
58600         return this.selections.length;
58601     },
58602
58603     /**
58604      * Selects the first row in the grid.
58605      */
58606     selectFirstRow : function(){
58607         this.selectRow(0);
58608     },
58609
58610     /**
58611      * Select the last row.
58612      * @param {Boolean} keepExisting (optional) True to keep existing selections
58613      */
58614     selectLastRow : function(keepExisting){
58615         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58616     },
58617
58618     /**
58619      * Selects the row immediately following the last selected row.
58620      * @param {Boolean} keepExisting (optional) True to keep existing selections
58621      */
58622     selectNext : function(keepExisting){
58623         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58624             this.selectRow(this.last+1, keepExisting);
58625             this.grid.getView().focusRow(this.last);
58626         }
58627     },
58628
58629     /**
58630      * Selects the row that precedes the last selected row.
58631      * @param {Boolean} keepExisting (optional) True to keep existing selections
58632      */
58633     selectPrevious : function(keepExisting){
58634         if(this.last){
58635             this.selectRow(this.last-1, keepExisting);
58636             this.grid.getView().focusRow(this.last);
58637         }
58638     },
58639
58640     /**
58641      * Returns the selected records
58642      * @return {Array} Array of selected records
58643      */
58644     getSelections : function(){
58645         return [].concat(this.selections.items);
58646     },
58647
58648     /**
58649      * Returns the first selected record.
58650      * @return {Record}
58651      */
58652     getSelected : function(){
58653         return this.selections.itemAt(0);
58654     },
58655
58656
58657     /**
58658      * Clears all selections.
58659      */
58660     clearSelections : function(fast){
58661         if(this.locked) {
58662             return;
58663         }
58664         if(fast !== true){
58665             var ds = this.grid.dataSource;
58666             var s = this.selections;
58667             s.each(function(r){
58668                 this.deselectRow(ds.indexOfId(r.id));
58669             }, this);
58670             s.clear();
58671         }else{
58672             this.selections.clear();
58673         }
58674         this.last = false;
58675     },
58676
58677
58678     /**
58679      * Selects all rows.
58680      */
58681     selectAll : function(){
58682         if(this.locked) {
58683             return;
58684         }
58685         this.selections.clear();
58686         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58687             this.selectRow(i, true);
58688         }
58689     },
58690
58691     /**
58692      * Returns True if there is a selection.
58693      * @return {Boolean}
58694      */
58695     hasSelection : function(){
58696         return this.selections.length > 0;
58697     },
58698
58699     /**
58700      * Returns True if the specified row is selected.
58701      * @param {Number/Record} record The record or index of the record to check
58702      * @return {Boolean}
58703      */
58704     isSelected : function(index){
58705         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58706         return (r && this.selections.key(r.id) ? true : false);
58707     },
58708
58709     /**
58710      * Returns True if the specified record id is selected.
58711      * @param {String} id The id of record to check
58712      * @return {Boolean}
58713      */
58714     isIdSelected : function(id){
58715         return (this.selections.key(id) ? true : false);
58716     },
58717
58718     // private
58719     handleMouseDown : function(e, t){
58720         var view = this.grid.getView(), rowIndex;
58721         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58722             return;
58723         };
58724         if(e.shiftKey && this.last !== false){
58725             var last = this.last;
58726             this.selectRange(last, rowIndex, e.ctrlKey);
58727             this.last = last; // reset the last
58728             view.focusRow(rowIndex);
58729         }else{
58730             var isSelected = this.isSelected(rowIndex);
58731             if(e.button !== 0 && isSelected){
58732                 view.focusRow(rowIndex);
58733             }else if(e.ctrlKey && isSelected){
58734                 this.deselectRow(rowIndex);
58735             }else if(!isSelected){
58736                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58737                 view.focusRow(rowIndex);
58738             }
58739         }
58740         this.fireEvent("afterselectionchange", this);
58741     },
58742     // private
58743     handleDragableRowClick :  function(grid, rowIndex, e) 
58744     {
58745         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58746             this.selectRow(rowIndex, false);
58747             grid.view.focusRow(rowIndex);
58748              this.fireEvent("afterselectionchange", this);
58749         }
58750     },
58751     
58752     /**
58753      * Selects multiple rows.
58754      * @param {Array} rows Array of the indexes of the row to select
58755      * @param {Boolean} keepExisting (optional) True to keep existing selections
58756      */
58757     selectRows : function(rows, keepExisting){
58758         if(!keepExisting){
58759             this.clearSelections();
58760         }
58761         for(var i = 0, len = rows.length; i < len; i++){
58762             this.selectRow(rows[i], true);
58763         }
58764     },
58765
58766     /**
58767      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58768      * @param {Number} startRow The index of the first row in the range
58769      * @param {Number} endRow The index of the last row in the range
58770      * @param {Boolean} keepExisting (optional) True to retain existing selections
58771      */
58772     selectRange : function(startRow, endRow, keepExisting){
58773         if(this.locked) {
58774             return;
58775         }
58776         if(!keepExisting){
58777             this.clearSelections();
58778         }
58779         if(startRow <= endRow){
58780             for(var i = startRow; i <= endRow; i++){
58781                 this.selectRow(i, true);
58782             }
58783         }else{
58784             for(var i = startRow; i >= endRow; i--){
58785                 this.selectRow(i, true);
58786             }
58787         }
58788     },
58789
58790     /**
58791      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58792      * @param {Number} startRow The index of the first row in the range
58793      * @param {Number} endRow The index of the last row in the range
58794      */
58795     deselectRange : function(startRow, endRow, preventViewNotify){
58796         if(this.locked) {
58797             return;
58798         }
58799         for(var i = startRow; i <= endRow; i++){
58800             this.deselectRow(i, preventViewNotify);
58801         }
58802     },
58803
58804     /**
58805      * Selects a row.
58806      * @param {Number} row The index of the row to select
58807      * @param {Boolean} keepExisting (optional) True to keep existing selections
58808      */
58809     selectRow : function(index, keepExisting, preventViewNotify){
58810         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58811             return;
58812         }
58813         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58814             if(!keepExisting || this.singleSelect){
58815                 this.clearSelections();
58816             }
58817             var r = this.grid.dataSource.getAt(index);
58818             this.selections.add(r);
58819             this.last = this.lastActive = index;
58820             if(!preventViewNotify){
58821                 this.grid.getView().onRowSelect(index);
58822             }
58823             this.fireEvent("rowselect", this, index, r);
58824             this.fireEvent("selectionchange", this);
58825         }
58826     },
58827
58828     /**
58829      * Deselects a row.
58830      * @param {Number} row The index of the row to deselect
58831      */
58832     deselectRow : function(index, preventViewNotify){
58833         if(this.locked) {
58834             return;
58835         }
58836         if(this.last == index){
58837             this.last = false;
58838         }
58839         if(this.lastActive == index){
58840             this.lastActive = false;
58841         }
58842         var r = this.grid.dataSource.getAt(index);
58843         this.selections.remove(r);
58844         if(!preventViewNotify){
58845             this.grid.getView().onRowDeselect(index);
58846         }
58847         this.fireEvent("rowdeselect", this, index);
58848         this.fireEvent("selectionchange", this);
58849     },
58850
58851     // private
58852     restoreLast : function(){
58853         if(this._last){
58854             this.last = this._last;
58855         }
58856     },
58857
58858     // private
58859     acceptsNav : function(row, col, cm){
58860         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58861     },
58862
58863     // private
58864     onEditorKey : function(field, e){
58865         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58866         if(k == e.TAB){
58867             e.stopEvent();
58868             ed.completeEdit();
58869             if(e.shiftKey){
58870                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58871             }else{
58872                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58873             }
58874         }else if(k == e.ENTER && !e.ctrlKey){
58875             e.stopEvent();
58876             ed.completeEdit();
58877             if(e.shiftKey){
58878                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58879             }else{
58880                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58881             }
58882         }else if(k == e.ESC){
58883             ed.cancelEdit();
58884         }
58885         if(newCell){
58886             g.startEditing(newCell[0], newCell[1]);
58887         }
58888     }
58889 });/*
58890  * Based on:
58891  * Ext JS Library 1.1.1
58892  * Copyright(c) 2006-2007, Ext JS, LLC.
58893  *
58894  * Originally Released Under LGPL - original licence link has changed is not relivant.
58895  *
58896  * Fork - LGPL
58897  * <script type="text/javascript">
58898  */
58899 /**
58900  * @class Roo.grid.CellSelectionModel
58901  * @extends Roo.grid.AbstractSelectionModel
58902  * This class provides the basic implementation for cell selection in a grid.
58903  * @constructor
58904  * @param {Object} config The object containing the configuration of this model.
58905  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58906  */
58907 Roo.grid.CellSelectionModel = function(config){
58908     Roo.apply(this, config);
58909
58910     this.selection = null;
58911
58912     this.addEvents({
58913         /**
58914              * @event beforerowselect
58915              * Fires before a cell is selected.
58916              * @param {SelectionModel} this
58917              * @param {Number} rowIndex The selected row index
58918              * @param {Number} colIndex The selected cell index
58919              */
58920             "beforecellselect" : true,
58921         /**
58922              * @event cellselect
58923              * Fires when a cell is selected.
58924              * @param {SelectionModel} this
58925              * @param {Number} rowIndex The selected row index
58926              * @param {Number} colIndex The selected cell index
58927              */
58928             "cellselect" : true,
58929         /**
58930              * @event selectionchange
58931              * Fires when the active selection changes.
58932              * @param {SelectionModel} this
58933              * @param {Object} selection null for no selection or an object (o) with two properties
58934                 <ul>
58935                 <li>o.record: the record object for the row the selection is in</li>
58936                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58937                 </ul>
58938              */
58939             "selectionchange" : true,
58940         /**
58941              * @event tabend
58942              * Fires when the tab (or enter) was pressed on the last editable cell
58943              * You can use this to trigger add new row.
58944              * @param {SelectionModel} this
58945              */
58946             "tabend" : true,
58947          /**
58948              * @event beforeeditnext
58949              * Fires before the next editable sell is made active
58950              * You can use this to skip to another cell or fire the tabend
58951              *    if you set cell to false
58952              * @param {Object} eventdata object : { cell : [ row, col ] } 
58953              */
58954             "beforeeditnext" : true
58955     });
58956     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58957 };
58958
58959 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58960     
58961     enter_is_tab: false,
58962
58963     /** @ignore */
58964     initEvents : function(){
58965         this.grid.on("mousedown", this.handleMouseDown, this);
58966         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58967         var view = this.grid.view;
58968         view.on("refresh", this.onViewChange, this);
58969         view.on("rowupdated", this.onRowUpdated, this);
58970         view.on("beforerowremoved", this.clearSelections, this);
58971         view.on("beforerowsinserted", this.clearSelections, this);
58972         if(this.grid.isEditor){
58973             this.grid.on("beforeedit", this.beforeEdit,  this);
58974         }
58975     },
58976
58977         //private
58978     beforeEdit : function(e){
58979         this.select(e.row, e.column, false, true, e.record);
58980     },
58981
58982         //private
58983     onRowUpdated : function(v, index, r){
58984         if(this.selection && this.selection.record == r){
58985             v.onCellSelect(index, this.selection.cell[1]);
58986         }
58987     },
58988
58989         //private
58990     onViewChange : function(){
58991         this.clearSelections(true);
58992     },
58993
58994         /**
58995          * Returns the currently selected cell,.
58996          * @return {Array} The selected cell (row, column) or null if none selected.
58997          */
58998     getSelectedCell : function(){
58999         return this.selection ? this.selection.cell : null;
59000     },
59001
59002     /**
59003      * Clears all selections.
59004      * @param {Boolean} true to prevent the gridview from being notified about the change.
59005      */
59006     clearSelections : function(preventNotify){
59007         var s = this.selection;
59008         if(s){
59009             if(preventNotify !== true){
59010                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
59011             }
59012             this.selection = null;
59013             this.fireEvent("selectionchange", this, null);
59014         }
59015     },
59016
59017     /**
59018      * Returns true if there is a selection.
59019      * @return {Boolean}
59020      */
59021     hasSelection : function(){
59022         return this.selection ? true : false;
59023     },
59024
59025     /** @ignore */
59026     handleMouseDown : function(e, t){
59027         var v = this.grid.getView();
59028         if(this.isLocked()){
59029             return;
59030         };
59031         var row = v.findRowIndex(t);
59032         var cell = v.findCellIndex(t);
59033         if(row !== false && cell !== false){
59034             this.select(row, cell);
59035         }
59036     },
59037
59038     /**
59039      * Selects a cell.
59040      * @param {Number} rowIndex
59041      * @param {Number} collIndex
59042      */
59043     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59044         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59045             this.clearSelections();
59046             r = r || this.grid.dataSource.getAt(rowIndex);
59047             this.selection = {
59048                 record : r,
59049                 cell : [rowIndex, colIndex]
59050             };
59051             if(!preventViewNotify){
59052                 var v = this.grid.getView();
59053                 v.onCellSelect(rowIndex, colIndex);
59054                 if(preventFocus !== true){
59055                     v.focusCell(rowIndex, colIndex);
59056                 }
59057             }
59058             this.fireEvent("cellselect", this, rowIndex, colIndex);
59059             this.fireEvent("selectionchange", this, this.selection);
59060         }
59061     },
59062
59063         //private
59064     isSelectable : function(rowIndex, colIndex, cm){
59065         return !cm.isHidden(colIndex);
59066     },
59067
59068     /** @ignore */
59069     handleKeyDown : function(e){
59070         //Roo.log('Cell Sel Model handleKeyDown');
59071         if(!e.isNavKeyPress()){
59072             return;
59073         }
59074         var g = this.grid, s = this.selection;
59075         if(!s){
59076             e.stopEvent();
59077             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59078             if(cell){
59079                 this.select(cell[0], cell[1]);
59080             }
59081             return;
59082         }
59083         var sm = this;
59084         var walk = function(row, col, step){
59085             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59086         };
59087         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59088         var newCell;
59089
59090       
59091
59092         switch(k){
59093             case e.TAB:
59094                 // handled by onEditorKey
59095                 if (g.isEditor && g.editing) {
59096                     return;
59097                 }
59098                 if(e.shiftKey) {
59099                     newCell = walk(r, c-1, -1);
59100                 } else {
59101                     newCell = walk(r, c+1, 1);
59102                 }
59103                 break;
59104             
59105             case e.DOWN:
59106                newCell = walk(r+1, c, 1);
59107                 break;
59108             
59109             case e.UP:
59110                 newCell = walk(r-1, c, -1);
59111                 break;
59112             
59113             case e.RIGHT:
59114                 newCell = walk(r, c+1, 1);
59115                 break;
59116             
59117             case e.LEFT:
59118                 newCell = walk(r, c-1, -1);
59119                 break;
59120             
59121             case e.ENTER:
59122                 
59123                 if(g.isEditor && !g.editing){
59124                    g.startEditing(r, c);
59125                    e.stopEvent();
59126                    return;
59127                 }
59128                 
59129                 
59130              break;
59131         };
59132         if(newCell){
59133             this.select(newCell[0], newCell[1]);
59134             e.stopEvent();
59135             
59136         }
59137     },
59138
59139     acceptsNav : function(row, col, cm){
59140         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59141     },
59142     /**
59143      * Selects a cell.
59144      * @param {Number} field (not used) - as it's normally used as a listener
59145      * @param {Number} e - event - fake it by using
59146      *
59147      * var e = Roo.EventObjectImpl.prototype;
59148      * e.keyCode = e.TAB
59149      *
59150      * 
59151      */
59152     onEditorKey : function(field, e){
59153         
59154         var k = e.getKey(),
59155             newCell,
59156             g = this.grid,
59157             ed = g.activeEditor,
59158             forward = false;
59159         ///Roo.log('onEditorKey' + k);
59160         
59161         
59162         if (this.enter_is_tab && k == e.ENTER) {
59163             k = e.TAB;
59164         }
59165         
59166         if(k == e.TAB){
59167             if(e.shiftKey){
59168                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59169             }else{
59170                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59171                 forward = true;
59172             }
59173             
59174             e.stopEvent();
59175             
59176         } else if(k == e.ENTER &&  !e.ctrlKey){
59177             ed.completeEdit();
59178             e.stopEvent();
59179             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59180         
59181                 } else if(k == e.ESC){
59182             ed.cancelEdit();
59183         }
59184                 
59185         if (newCell) {
59186             var ecall = { cell : newCell, forward : forward };
59187             this.fireEvent('beforeeditnext', ecall );
59188             newCell = ecall.cell;
59189                         forward = ecall.forward;
59190         }
59191                 
59192         if(newCell){
59193             //Roo.log('next cell after edit');
59194             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59195         } else if (forward) {
59196             // tabbed past last
59197             this.fireEvent.defer(100, this, ['tabend',this]);
59198         }
59199     }
59200 });/*
59201  * Based on:
59202  * Ext JS Library 1.1.1
59203  * Copyright(c) 2006-2007, Ext JS, LLC.
59204  *
59205  * Originally Released Under LGPL - original licence link has changed is not relivant.
59206  *
59207  * Fork - LGPL
59208  * <script type="text/javascript">
59209  */
59210  
59211 /**
59212  * @class Roo.grid.EditorGrid
59213  * @extends Roo.grid.Grid
59214  * Class for creating and editable grid.
59215  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59216  * The container MUST have some type of size defined for the grid to fill. The container will be 
59217  * automatically set to position relative if it isn't already.
59218  * @param {Object} dataSource The data model to bind to
59219  * @param {Object} colModel The column model with info about this grid's columns
59220  */
59221 Roo.grid.EditorGrid = function(container, config){
59222     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59223     this.getGridEl().addClass("xedit-grid");
59224
59225     if(!this.selModel){
59226         this.selModel = new Roo.grid.CellSelectionModel();
59227     }
59228
59229     this.activeEditor = null;
59230
59231         this.addEvents({
59232             /**
59233              * @event beforeedit
59234              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59235              * <ul style="padding:5px;padding-left:16px;">
59236              * <li>grid - This grid</li>
59237              * <li>record - The record being edited</li>
59238              * <li>field - The field name being edited</li>
59239              * <li>value - The value for the field being edited.</li>
59240              * <li>row - The grid row index</li>
59241              * <li>column - The grid column index</li>
59242              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59243              * </ul>
59244              * @param {Object} e An edit event (see above for description)
59245              */
59246             "beforeedit" : true,
59247             /**
59248              * @event afteredit
59249              * Fires after a cell is edited. <br />
59250              * <ul style="padding:5px;padding-left:16px;">
59251              * <li>grid - This grid</li>
59252              * <li>record - The record being edited</li>
59253              * <li>field - The field name being edited</li>
59254              * <li>value - The value being set</li>
59255              * <li>originalValue - The original value for the field, before the edit.</li>
59256              * <li>row - The grid row index</li>
59257              * <li>column - The grid column index</li>
59258              * </ul>
59259              * @param {Object} e An edit event (see above for description)
59260              */
59261             "afteredit" : true,
59262             /**
59263              * @event validateedit
59264              * Fires after a cell is edited, but before the value is set in the record. 
59265          * You can use this to modify the value being set in the field, Return false
59266              * to cancel the change. The edit event object has the following properties <br />
59267              * <ul style="padding:5px;padding-left:16px;">
59268          * <li>editor - This editor</li>
59269              * <li>grid - This grid</li>
59270              * <li>record - The record being edited</li>
59271              * <li>field - The field name being edited</li>
59272              * <li>value - The value being set</li>
59273              * <li>originalValue - The original value for the field, before the edit.</li>
59274              * <li>row - The grid row index</li>
59275              * <li>column - The grid column index</li>
59276              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59277              * </ul>
59278              * @param {Object} e An edit event (see above for description)
59279              */
59280             "validateedit" : true
59281         });
59282     this.on("bodyscroll", this.stopEditing,  this);
59283     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59284 };
59285
59286 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59287     /**
59288      * @cfg {Number} clicksToEdit
59289      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59290      */
59291     clicksToEdit: 2,
59292
59293     // private
59294     isEditor : true,
59295     // private
59296     trackMouseOver: false, // causes very odd FF errors
59297
59298     onCellDblClick : function(g, row, col){
59299         this.startEditing(row, col);
59300     },
59301
59302     onEditComplete : function(ed, value, startValue){
59303         this.editing = false;
59304         this.activeEditor = null;
59305         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59306         var r = ed.record;
59307         var field = this.colModel.getDataIndex(ed.col);
59308         var e = {
59309             grid: this,
59310             record: r,
59311             field: field,
59312             originalValue: startValue,
59313             value: value,
59314             row: ed.row,
59315             column: ed.col,
59316             cancel:false,
59317             editor: ed
59318         };
59319         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59320         cell.show();
59321           
59322         if(String(value) !== String(startValue)){
59323             
59324             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59325                 r.set(field, e.value);
59326                 // if we are dealing with a combo box..
59327                 // then we also set the 'name' colum to be the displayField
59328                 if (ed.field.displayField && ed.field.name) {
59329                     r.set(ed.field.name, ed.field.el.dom.value);
59330                 }
59331                 
59332                 delete e.cancel; //?? why!!!
59333                 this.fireEvent("afteredit", e);
59334             }
59335         } else {
59336             this.fireEvent("afteredit", e); // always fire it!
59337         }
59338         this.view.focusCell(ed.row, ed.col);
59339     },
59340
59341     /**
59342      * Starts editing the specified for the specified row/column
59343      * @param {Number} rowIndex
59344      * @param {Number} colIndex
59345      */
59346     startEditing : function(row, col){
59347         this.stopEditing();
59348         if(this.colModel.isCellEditable(col, row)){
59349             this.view.ensureVisible(row, col, true);
59350           
59351             var r = this.dataSource.getAt(row);
59352             var field = this.colModel.getDataIndex(col);
59353             var cell = Roo.get(this.view.getCell(row,col));
59354             var e = {
59355                 grid: this,
59356                 record: r,
59357                 field: field,
59358                 value: r.data[field],
59359                 row: row,
59360                 column: col,
59361                 cancel:false 
59362             };
59363             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59364                 this.editing = true;
59365                 var ed = this.colModel.getCellEditor(col, row);
59366                 
59367                 if (!ed) {
59368                     return;
59369                 }
59370                 if(!ed.rendered){
59371                     ed.render(ed.parentEl || document.body);
59372                 }
59373                 ed.field.reset();
59374                
59375                 cell.hide();
59376                 
59377                 (function(){ // complex but required for focus issues in safari, ie and opera
59378                     ed.row = row;
59379                     ed.col = col;
59380                     ed.record = r;
59381                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59382                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59383                     this.activeEditor = ed;
59384                     var v = r.data[field];
59385                     ed.startEdit(this.view.getCell(row, col), v);
59386                     // combo's with 'displayField and name set
59387                     if (ed.field.displayField && ed.field.name) {
59388                         ed.field.el.dom.value = r.data[ed.field.name];
59389                     }
59390                     
59391                     
59392                 }).defer(50, this);
59393             }
59394         }
59395     },
59396         
59397     /**
59398      * Stops any active editing
59399      */
59400     stopEditing : function(){
59401         if(this.activeEditor){
59402             this.activeEditor.completeEdit();
59403         }
59404         this.activeEditor = null;
59405     },
59406         
59407          /**
59408      * Called to get grid's drag proxy text, by default returns this.ddText.
59409      * @return {String}
59410      */
59411     getDragDropText : function(){
59412         var count = this.selModel.getSelectedCell() ? 1 : 0;
59413         return String.format(this.ddText, count, count == 1 ? '' : 's');
59414     }
59415         
59416 });/*
59417  * Based on:
59418  * Ext JS Library 1.1.1
59419  * Copyright(c) 2006-2007, Ext JS, LLC.
59420  *
59421  * Originally Released Under LGPL - original licence link has changed is not relivant.
59422  *
59423  * Fork - LGPL
59424  * <script type="text/javascript">
59425  */
59426
59427 // private - not really -- you end up using it !
59428 // This is a support class used internally by the Grid components
59429
59430 /**
59431  * @class Roo.grid.GridEditor
59432  * @extends Roo.Editor
59433  * Class for creating and editable grid elements.
59434  * @param {Object} config any settings (must include field)
59435  */
59436 Roo.grid.GridEditor = function(field, config){
59437     if (!config && field.field) {
59438         config = field;
59439         field = Roo.factory(config.field, Roo.form);
59440     }
59441     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59442     field.monitorTab = false;
59443 };
59444
59445 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59446     
59447     /**
59448      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59449      */
59450     
59451     alignment: "tl-tl",
59452     autoSize: "width",
59453     hideEl : false,
59454     cls: "x-small-editor x-grid-editor",
59455     shim:false,
59456     shadow:"frame"
59457 });/*
59458  * Based on:
59459  * Ext JS Library 1.1.1
59460  * Copyright(c) 2006-2007, Ext JS, LLC.
59461  *
59462  * Originally Released Under LGPL - original licence link has changed is not relivant.
59463  *
59464  * Fork - LGPL
59465  * <script type="text/javascript">
59466  */
59467   
59468
59469   
59470 Roo.grid.PropertyRecord = Roo.data.Record.create([
59471     {name:'name',type:'string'},  'value'
59472 ]);
59473
59474
59475 Roo.grid.PropertyStore = function(grid, source){
59476     this.grid = grid;
59477     this.store = new Roo.data.Store({
59478         recordType : Roo.grid.PropertyRecord
59479     });
59480     this.store.on('update', this.onUpdate,  this);
59481     if(source){
59482         this.setSource(source);
59483     }
59484     Roo.grid.PropertyStore.superclass.constructor.call(this);
59485 };
59486
59487
59488
59489 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59490     setSource : function(o){
59491         this.source = o;
59492         this.store.removeAll();
59493         var data = [];
59494         for(var k in o){
59495             if(this.isEditableValue(o[k])){
59496                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59497             }
59498         }
59499         this.store.loadRecords({records: data}, {}, true);
59500     },
59501
59502     onUpdate : function(ds, record, type){
59503         if(type == Roo.data.Record.EDIT){
59504             var v = record.data['value'];
59505             var oldValue = record.modified['value'];
59506             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59507                 this.source[record.id] = v;
59508                 record.commit();
59509                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59510             }else{
59511                 record.reject();
59512             }
59513         }
59514     },
59515
59516     getProperty : function(row){
59517        return this.store.getAt(row);
59518     },
59519
59520     isEditableValue: function(val){
59521         if(val && val instanceof Date){
59522             return true;
59523         }else if(typeof val == 'object' || typeof val == 'function'){
59524             return false;
59525         }
59526         return true;
59527     },
59528
59529     setValue : function(prop, value){
59530         this.source[prop] = value;
59531         this.store.getById(prop).set('value', value);
59532     },
59533
59534     getSource : function(){
59535         return this.source;
59536     }
59537 });
59538
59539 Roo.grid.PropertyColumnModel = function(grid, store){
59540     this.grid = grid;
59541     var g = Roo.grid;
59542     g.PropertyColumnModel.superclass.constructor.call(this, [
59543         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59544         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59545     ]);
59546     this.store = store;
59547     this.bselect = Roo.DomHelper.append(document.body, {
59548         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59549             {tag: 'option', value: 'true', html: 'true'},
59550             {tag: 'option', value: 'false', html: 'false'}
59551         ]
59552     });
59553     Roo.id(this.bselect);
59554     var f = Roo.form;
59555     this.editors = {
59556         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59557         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59558         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59559         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59560         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59561     };
59562     this.renderCellDelegate = this.renderCell.createDelegate(this);
59563     this.renderPropDelegate = this.renderProp.createDelegate(this);
59564 };
59565
59566 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59567     
59568     
59569     nameText : 'Name',
59570     valueText : 'Value',
59571     
59572     dateFormat : 'm/j/Y',
59573     
59574     
59575     renderDate : function(dateVal){
59576         return dateVal.dateFormat(this.dateFormat);
59577     },
59578
59579     renderBool : function(bVal){
59580         return bVal ? 'true' : 'false';
59581     },
59582
59583     isCellEditable : function(colIndex, rowIndex){
59584         return colIndex == 1;
59585     },
59586
59587     getRenderer : function(col){
59588         return col == 1 ?
59589             this.renderCellDelegate : this.renderPropDelegate;
59590     },
59591
59592     renderProp : function(v){
59593         return this.getPropertyName(v);
59594     },
59595
59596     renderCell : function(val){
59597         var rv = val;
59598         if(val instanceof Date){
59599             rv = this.renderDate(val);
59600         }else if(typeof val == 'boolean'){
59601             rv = this.renderBool(val);
59602         }
59603         return Roo.util.Format.htmlEncode(rv);
59604     },
59605
59606     getPropertyName : function(name){
59607         var pn = this.grid.propertyNames;
59608         return pn && pn[name] ? pn[name] : name;
59609     },
59610
59611     getCellEditor : function(colIndex, rowIndex){
59612         var p = this.store.getProperty(rowIndex);
59613         var n = p.data['name'], val = p.data['value'];
59614         
59615         if(typeof(this.grid.customEditors[n]) == 'string'){
59616             return this.editors[this.grid.customEditors[n]];
59617         }
59618         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59619             return this.grid.customEditors[n];
59620         }
59621         if(val instanceof Date){
59622             return this.editors['date'];
59623         }else if(typeof val == 'number'){
59624             return this.editors['number'];
59625         }else if(typeof val == 'boolean'){
59626             return this.editors['boolean'];
59627         }else{
59628             return this.editors['string'];
59629         }
59630     }
59631 });
59632
59633 /**
59634  * @class Roo.grid.PropertyGrid
59635  * @extends Roo.grid.EditorGrid
59636  * This class represents the  interface of a component based property grid control.
59637  * <br><br>Usage:<pre><code>
59638  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59639       
59640  });
59641  // set any options
59642  grid.render();
59643  * </code></pre>
59644   
59645  * @constructor
59646  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59647  * The container MUST have some type of size defined for the grid to fill. The container will be
59648  * automatically set to position relative if it isn't already.
59649  * @param {Object} config A config object that sets properties on this grid.
59650  */
59651 Roo.grid.PropertyGrid = function(container, config){
59652     config = config || {};
59653     var store = new Roo.grid.PropertyStore(this);
59654     this.store = store;
59655     var cm = new Roo.grid.PropertyColumnModel(this, store);
59656     store.store.sort('name', 'ASC');
59657     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59658         ds: store.store,
59659         cm: cm,
59660         enableColLock:false,
59661         enableColumnMove:false,
59662         stripeRows:false,
59663         trackMouseOver: false,
59664         clicksToEdit:1
59665     }, config));
59666     this.getGridEl().addClass('x-props-grid');
59667     this.lastEditRow = null;
59668     this.on('columnresize', this.onColumnResize, this);
59669     this.addEvents({
59670          /**
59671              * @event beforepropertychange
59672              * Fires before a property changes (return false to stop?)
59673              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59674              * @param {String} id Record Id
59675              * @param {String} newval New Value
59676          * @param {String} oldval Old Value
59677              */
59678         "beforepropertychange": true,
59679         /**
59680              * @event propertychange
59681              * Fires after a property changes
59682              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59683              * @param {String} id Record Id
59684              * @param {String} newval New Value
59685          * @param {String} oldval Old Value
59686              */
59687         "propertychange": true
59688     });
59689     this.customEditors = this.customEditors || {};
59690 };
59691 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59692     
59693      /**
59694      * @cfg {Object} customEditors map of colnames=> custom editors.
59695      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59696      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59697      * false disables editing of the field.
59698          */
59699     
59700       /**
59701      * @cfg {Object} propertyNames map of property Names to their displayed value
59702          */
59703     
59704     render : function(){
59705         Roo.grid.PropertyGrid.superclass.render.call(this);
59706         this.autoSize.defer(100, this);
59707     },
59708
59709     autoSize : function(){
59710         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59711         if(this.view){
59712             this.view.fitColumns();
59713         }
59714     },
59715
59716     onColumnResize : function(){
59717         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59718         this.autoSize();
59719     },
59720     /**
59721      * Sets the data for the Grid
59722      * accepts a Key => Value object of all the elements avaiable.
59723      * @param {Object} data  to appear in grid.
59724      */
59725     setSource : function(source){
59726         this.store.setSource(source);
59727         //this.autoSize();
59728     },
59729     /**
59730      * Gets all the data from the grid.
59731      * @return {Object} data  data stored in grid
59732      */
59733     getSource : function(){
59734         return this.store.getSource();
59735     }
59736 });/*
59737   
59738  * Licence LGPL
59739  
59740  */
59741  
59742 /**
59743  * @class Roo.grid.Calendar
59744  * @extends Roo.util.Grid
59745  * This class extends the Grid to provide a calendar widget
59746  * <br><br>Usage:<pre><code>
59747  var grid = new Roo.grid.Calendar("my-container-id", {
59748      ds: myDataStore,
59749      cm: myColModel,
59750      selModel: mySelectionModel,
59751      autoSizeColumns: true,
59752      monitorWindowResize: false,
59753      trackMouseOver: true
59754      eventstore : real data store..
59755  });
59756  // set any options
59757  grid.render();
59758   
59759   * @constructor
59760  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59761  * The container MUST have some type of size defined for the grid to fill. The container will be
59762  * automatically set to position relative if it isn't already.
59763  * @param {Object} config A config object that sets properties on this grid.
59764  */
59765 Roo.grid.Calendar = function(container, config){
59766         // initialize the container
59767         this.container = Roo.get(container);
59768         this.container.update("");
59769         this.container.setStyle("overflow", "hidden");
59770     this.container.addClass('x-grid-container');
59771
59772     this.id = this.container.id;
59773
59774     Roo.apply(this, config);
59775     // check and correct shorthanded configs
59776     
59777     var rows = [];
59778     var d =1;
59779     for (var r = 0;r < 6;r++) {
59780         
59781         rows[r]=[];
59782         for (var c =0;c < 7;c++) {
59783             rows[r][c]= '';
59784         }
59785     }
59786     if (this.eventStore) {
59787         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59788         this.eventStore.on('load',this.onLoad, this);
59789         this.eventStore.on('beforeload',this.clearEvents, this);
59790          
59791     }
59792     
59793     this.dataSource = new Roo.data.Store({
59794             proxy: new Roo.data.MemoryProxy(rows),
59795             reader: new Roo.data.ArrayReader({}, [
59796                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59797     });
59798
59799     this.dataSource.load();
59800     this.ds = this.dataSource;
59801     this.ds.xmodule = this.xmodule || false;
59802     
59803     
59804     var cellRender = function(v,x,r)
59805     {
59806         return String.format(
59807             '<div class="fc-day  fc-widget-content"><div>' +
59808                 '<div class="fc-event-container"></div>' +
59809                 '<div class="fc-day-number">{0}</div>'+
59810                 
59811                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59812             '</div></div>', v);
59813     
59814     }
59815     
59816     
59817     this.colModel = new Roo.grid.ColumnModel( [
59818         {
59819             xtype: 'ColumnModel',
59820             xns: Roo.grid,
59821             dataIndex : 'weekday0',
59822             header : 'Sunday',
59823             renderer : cellRender
59824         },
59825         {
59826             xtype: 'ColumnModel',
59827             xns: Roo.grid,
59828             dataIndex : 'weekday1',
59829             header : 'Monday',
59830             renderer : cellRender
59831         },
59832         {
59833             xtype: 'ColumnModel',
59834             xns: Roo.grid,
59835             dataIndex : 'weekday2',
59836             header : 'Tuesday',
59837             renderer : cellRender
59838         },
59839         {
59840             xtype: 'ColumnModel',
59841             xns: Roo.grid,
59842             dataIndex : 'weekday3',
59843             header : 'Wednesday',
59844             renderer : cellRender
59845         },
59846         {
59847             xtype: 'ColumnModel',
59848             xns: Roo.grid,
59849             dataIndex : 'weekday4',
59850             header : 'Thursday',
59851             renderer : cellRender
59852         },
59853         {
59854             xtype: 'ColumnModel',
59855             xns: Roo.grid,
59856             dataIndex : 'weekday5',
59857             header : 'Friday',
59858             renderer : cellRender
59859         },
59860         {
59861             xtype: 'ColumnModel',
59862             xns: Roo.grid,
59863             dataIndex : 'weekday6',
59864             header : 'Saturday',
59865             renderer : cellRender
59866         }
59867     ]);
59868     this.cm = this.colModel;
59869     this.cm.xmodule = this.xmodule || false;
59870  
59871         
59872           
59873     //this.selModel = new Roo.grid.CellSelectionModel();
59874     //this.sm = this.selModel;
59875     //this.selModel.init(this);
59876     
59877     
59878     if(this.width){
59879         this.container.setWidth(this.width);
59880     }
59881
59882     if(this.height){
59883         this.container.setHeight(this.height);
59884     }
59885     /** @private */
59886         this.addEvents({
59887         // raw events
59888         /**
59889          * @event click
59890          * The raw click event for the entire grid.
59891          * @param {Roo.EventObject} e
59892          */
59893         "click" : true,
59894         /**
59895          * @event dblclick
59896          * The raw dblclick event for the entire grid.
59897          * @param {Roo.EventObject} e
59898          */
59899         "dblclick" : true,
59900         /**
59901          * @event contextmenu
59902          * The raw contextmenu event for the entire grid.
59903          * @param {Roo.EventObject} e
59904          */
59905         "contextmenu" : true,
59906         /**
59907          * @event mousedown
59908          * The raw mousedown event for the entire grid.
59909          * @param {Roo.EventObject} e
59910          */
59911         "mousedown" : true,
59912         /**
59913          * @event mouseup
59914          * The raw mouseup event for the entire grid.
59915          * @param {Roo.EventObject} e
59916          */
59917         "mouseup" : true,
59918         /**
59919          * @event mouseover
59920          * The raw mouseover event for the entire grid.
59921          * @param {Roo.EventObject} e
59922          */
59923         "mouseover" : true,
59924         /**
59925          * @event mouseout
59926          * The raw mouseout event for the entire grid.
59927          * @param {Roo.EventObject} e
59928          */
59929         "mouseout" : true,
59930         /**
59931          * @event keypress
59932          * The raw keypress event for the entire grid.
59933          * @param {Roo.EventObject} e
59934          */
59935         "keypress" : true,
59936         /**
59937          * @event keydown
59938          * The raw keydown event for the entire grid.
59939          * @param {Roo.EventObject} e
59940          */
59941         "keydown" : true,
59942
59943         // custom events
59944
59945         /**
59946          * @event cellclick
59947          * Fires when a cell is clicked
59948          * @param {Grid} this
59949          * @param {Number} rowIndex
59950          * @param {Number} columnIndex
59951          * @param {Roo.EventObject} e
59952          */
59953         "cellclick" : true,
59954         /**
59955          * @event celldblclick
59956          * Fires when a cell is double clicked
59957          * @param {Grid} this
59958          * @param {Number} rowIndex
59959          * @param {Number} columnIndex
59960          * @param {Roo.EventObject} e
59961          */
59962         "celldblclick" : true,
59963         /**
59964          * @event rowclick
59965          * Fires when a row is clicked
59966          * @param {Grid} this
59967          * @param {Number} rowIndex
59968          * @param {Roo.EventObject} e
59969          */
59970         "rowclick" : true,
59971         /**
59972          * @event rowdblclick
59973          * Fires when a row is double clicked
59974          * @param {Grid} this
59975          * @param {Number} rowIndex
59976          * @param {Roo.EventObject} e
59977          */
59978         "rowdblclick" : true,
59979         /**
59980          * @event headerclick
59981          * Fires when a header is clicked
59982          * @param {Grid} this
59983          * @param {Number} columnIndex
59984          * @param {Roo.EventObject} e
59985          */
59986         "headerclick" : true,
59987         /**
59988          * @event headerdblclick
59989          * Fires when a header cell is double clicked
59990          * @param {Grid} this
59991          * @param {Number} columnIndex
59992          * @param {Roo.EventObject} e
59993          */
59994         "headerdblclick" : true,
59995         /**
59996          * @event rowcontextmenu
59997          * Fires when a row is right clicked
59998          * @param {Grid} this
59999          * @param {Number} rowIndex
60000          * @param {Roo.EventObject} e
60001          */
60002         "rowcontextmenu" : true,
60003         /**
60004          * @event cellcontextmenu
60005          * Fires when a cell is right clicked
60006          * @param {Grid} this
60007          * @param {Number} rowIndex
60008          * @param {Number} cellIndex
60009          * @param {Roo.EventObject} e
60010          */
60011          "cellcontextmenu" : true,
60012         /**
60013          * @event headercontextmenu
60014          * Fires when a header is right clicked
60015          * @param {Grid} this
60016          * @param {Number} columnIndex
60017          * @param {Roo.EventObject} e
60018          */
60019         "headercontextmenu" : true,
60020         /**
60021          * @event bodyscroll
60022          * Fires when the body element is scrolled
60023          * @param {Number} scrollLeft
60024          * @param {Number} scrollTop
60025          */
60026         "bodyscroll" : true,
60027         /**
60028          * @event columnresize
60029          * Fires when the user resizes a column
60030          * @param {Number} columnIndex
60031          * @param {Number} newSize
60032          */
60033         "columnresize" : true,
60034         /**
60035          * @event columnmove
60036          * Fires when the user moves a column
60037          * @param {Number} oldIndex
60038          * @param {Number} newIndex
60039          */
60040         "columnmove" : true,
60041         /**
60042          * @event startdrag
60043          * Fires when row(s) start being dragged
60044          * @param {Grid} this
60045          * @param {Roo.GridDD} dd The drag drop object
60046          * @param {event} e The raw browser event
60047          */
60048         "startdrag" : true,
60049         /**
60050          * @event enddrag
60051          * Fires when a drag operation is complete
60052          * @param {Grid} this
60053          * @param {Roo.GridDD} dd The drag drop object
60054          * @param {event} e The raw browser event
60055          */
60056         "enddrag" : true,
60057         /**
60058          * @event dragdrop
60059          * Fires when dragged row(s) are dropped on a valid DD target
60060          * @param {Grid} this
60061          * @param {Roo.GridDD} dd The drag drop object
60062          * @param {String} targetId The target drag drop object
60063          * @param {event} e The raw browser event
60064          */
60065         "dragdrop" : true,
60066         /**
60067          * @event dragover
60068          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60069          * @param {Grid} this
60070          * @param {Roo.GridDD} dd The drag drop object
60071          * @param {String} targetId The target drag drop object
60072          * @param {event} e The raw browser event
60073          */
60074         "dragover" : true,
60075         /**
60076          * @event dragenter
60077          *  Fires when the dragged row(s) first cross another DD target while being dragged
60078          * @param {Grid} this
60079          * @param {Roo.GridDD} dd The drag drop object
60080          * @param {String} targetId The target drag drop object
60081          * @param {event} e The raw browser event
60082          */
60083         "dragenter" : true,
60084         /**
60085          * @event dragout
60086          * Fires when the dragged row(s) leave another DD target while being dragged
60087          * @param {Grid} this
60088          * @param {Roo.GridDD} dd The drag drop object
60089          * @param {String} targetId The target drag drop object
60090          * @param {event} e The raw browser event
60091          */
60092         "dragout" : true,
60093         /**
60094          * @event rowclass
60095          * Fires when a row is rendered, so you can change add a style to it.
60096          * @param {GridView} gridview   The grid view
60097          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60098          */
60099         'rowclass' : true,
60100
60101         /**
60102          * @event render
60103          * Fires when the grid is rendered
60104          * @param {Grid} grid
60105          */
60106         'render' : true,
60107             /**
60108              * @event select
60109              * Fires when a date is selected
60110              * @param {DatePicker} this
60111              * @param {Date} date The selected date
60112              */
60113         'select': true,
60114         /**
60115              * @event monthchange
60116              * Fires when the displayed month changes 
60117              * @param {DatePicker} this
60118              * @param {Date} date The selected month
60119              */
60120         'monthchange': true,
60121         /**
60122              * @event evententer
60123              * Fires when mouse over an event
60124              * @param {Calendar} this
60125              * @param {event} Event
60126              */
60127         'evententer': true,
60128         /**
60129              * @event eventleave
60130              * Fires when the mouse leaves an
60131              * @param {Calendar} this
60132              * @param {event}
60133              */
60134         'eventleave': true,
60135         /**
60136              * @event eventclick
60137              * Fires when the mouse click an
60138              * @param {Calendar} this
60139              * @param {event}
60140              */
60141         'eventclick': true,
60142         /**
60143              * @event eventrender
60144              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60145              * @param {Calendar} this
60146              * @param {data} data to be modified
60147              */
60148         'eventrender': true
60149         
60150     });
60151
60152     Roo.grid.Grid.superclass.constructor.call(this);
60153     this.on('render', function() {
60154         this.view.el.addClass('x-grid-cal'); 
60155         
60156         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60157
60158     },this);
60159     
60160     if (!Roo.grid.Calendar.style) {
60161         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60162             
60163             
60164             '.x-grid-cal .x-grid-col' :  {
60165                 height: 'auto !important',
60166                 'vertical-align': 'top'
60167             },
60168             '.x-grid-cal  .fc-event-hori' : {
60169                 height: '14px'
60170             }
60171              
60172             
60173         }, Roo.id());
60174     }
60175
60176     
60177     
60178 };
60179 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60180     /**
60181      * @cfg {Store} eventStore The store that loads events.
60182      */
60183     eventStore : 25,
60184
60185      
60186     activeDate : false,
60187     startDay : 0,
60188     autoWidth : true,
60189     monitorWindowResize : false,
60190
60191     
60192     resizeColumns : function() {
60193         var col = (this.view.el.getWidth() / 7) - 3;
60194         // loop through cols, and setWidth
60195         for(var i =0 ; i < 7 ; i++){
60196             this.cm.setColumnWidth(i, col);
60197         }
60198     },
60199      setDate :function(date) {
60200         
60201         Roo.log('setDate?');
60202         
60203         this.resizeColumns();
60204         var vd = this.activeDate;
60205         this.activeDate = date;
60206 //        if(vd && this.el){
60207 //            var t = date.getTime();
60208 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60209 //                Roo.log('using add remove');
60210 //                
60211 //                this.fireEvent('monthchange', this, date);
60212 //                
60213 //                this.cells.removeClass("fc-state-highlight");
60214 //                this.cells.each(function(c){
60215 //                   if(c.dateValue == t){
60216 //                       c.addClass("fc-state-highlight");
60217 //                       setTimeout(function(){
60218 //                            try{c.dom.firstChild.focus();}catch(e){}
60219 //                       }, 50);
60220 //                       return false;
60221 //                   }
60222 //                   return true;
60223 //                });
60224 //                return;
60225 //            }
60226 //        }
60227         
60228         var days = date.getDaysInMonth();
60229         
60230         var firstOfMonth = date.getFirstDateOfMonth();
60231         var startingPos = firstOfMonth.getDay()-this.startDay;
60232         
60233         if(startingPos < this.startDay){
60234             startingPos += 7;
60235         }
60236         
60237         var pm = date.add(Date.MONTH, -1);
60238         var prevStart = pm.getDaysInMonth()-startingPos;
60239 //        
60240         
60241         
60242         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60243         
60244         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60245         //this.cells.addClassOnOver('fc-state-hover');
60246         
60247         var cells = this.cells.elements;
60248         var textEls = this.textNodes;
60249         
60250         //Roo.each(cells, function(cell){
60251         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60252         //});
60253         
60254         days += startingPos;
60255
60256         // convert everything to numbers so it's fast
60257         var day = 86400000;
60258         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60259         //Roo.log(d);
60260         //Roo.log(pm);
60261         //Roo.log(prevStart);
60262         
60263         var today = new Date().clearTime().getTime();
60264         var sel = date.clearTime().getTime();
60265         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60266         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60267         var ddMatch = this.disabledDatesRE;
60268         var ddText = this.disabledDatesText;
60269         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60270         var ddaysText = this.disabledDaysText;
60271         var format = this.format;
60272         
60273         var setCellClass = function(cal, cell){
60274             
60275             //Roo.log('set Cell Class');
60276             cell.title = "";
60277             var t = d.getTime();
60278             
60279             //Roo.log(d);
60280             
60281             
60282             cell.dateValue = t;
60283             if(t == today){
60284                 cell.className += " fc-today";
60285                 cell.className += " fc-state-highlight";
60286                 cell.title = cal.todayText;
60287             }
60288             if(t == sel){
60289                 // disable highlight in other month..
60290                 cell.className += " fc-state-highlight";
60291                 
60292             }
60293             // disabling
60294             if(t < min) {
60295                 //cell.className = " fc-state-disabled";
60296                 cell.title = cal.minText;
60297                 return;
60298             }
60299             if(t > max) {
60300                 //cell.className = " fc-state-disabled";
60301                 cell.title = cal.maxText;
60302                 return;
60303             }
60304             if(ddays){
60305                 if(ddays.indexOf(d.getDay()) != -1){
60306                     // cell.title = ddaysText;
60307                    // cell.className = " fc-state-disabled";
60308                 }
60309             }
60310             if(ddMatch && format){
60311                 var fvalue = d.dateFormat(format);
60312                 if(ddMatch.test(fvalue)){
60313                     cell.title = ddText.replace("%0", fvalue);
60314                    cell.className = " fc-state-disabled";
60315                 }
60316             }
60317             
60318             if (!cell.initialClassName) {
60319                 cell.initialClassName = cell.dom.className;
60320             }
60321             
60322             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60323         };
60324
60325         var i = 0;
60326         
60327         for(; i < startingPos; i++) {
60328             cells[i].dayName =  (++prevStart);
60329             Roo.log(textEls[i]);
60330             d.setDate(d.getDate()+1);
60331             
60332             //cells[i].className = "fc-past fc-other-month";
60333             setCellClass(this, cells[i]);
60334         }
60335         
60336         var intDay = 0;
60337         
60338         for(; i < days; i++){
60339             intDay = i - startingPos + 1;
60340             cells[i].dayName =  (intDay);
60341             d.setDate(d.getDate()+1);
60342             
60343             cells[i].className = ''; // "x-date-active";
60344             setCellClass(this, cells[i]);
60345         }
60346         var extraDays = 0;
60347         
60348         for(; i < 42; i++) {
60349             //textEls[i].innerHTML = (++extraDays);
60350             
60351             d.setDate(d.getDate()+1);
60352             cells[i].dayName = (++extraDays);
60353             cells[i].className = "fc-future fc-other-month";
60354             setCellClass(this, cells[i]);
60355         }
60356         
60357         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60358         
60359         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60360         
60361         // this will cause all the cells to mis
60362         var rows= [];
60363         var i =0;
60364         for (var r = 0;r < 6;r++) {
60365             for (var c =0;c < 7;c++) {
60366                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60367             }    
60368         }
60369         
60370         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60371         for(i=0;i<cells.length;i++) {
60372             
60373             this.cells.elements[i].dayName = cells[i].dayName ;
60374             this.cells.elements[i].className = cells[i].className;
60375             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60376             this.cells.elements[i].title = cells[i].title ;
60377             this.cells.elements[i].dateValue = cells[i].dateValue ;
60378         }
60379         
60380         
60381         
60382         
60383         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60384         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60385         
60386         ////if(totalRows != 6){
60387             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60388            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60389        // }
60390         
60391         this.fireEvent('monthchange', this, date);
60392         
60393         
60394     },
60395  /**
60396      * Returns the grid's SelectionModel.
60397      * @return {SelectionModel}
60398      */
60399     getSelectionModel : function(){
60400         if(!this.selModel){
60401             this.selModel = new Roo.grid.CellSelectionModel();
60402         }
60403         return this.selModel;
60404     },
60405
60406     load: function() {
60407         this.eventStore.load()
60408         
60409         
60410         
60411     },
60412     
60413     findCell : function(dt) {
60414         dt = dt.clearTime().getTime();
60415         var ret = false;
60416         this.cells.each(function(c){
60417             //Roo.log("check " +c.dateValue + '?=' + dt);
60418             if(c.dateValue == dt){
60419                 ret = c;
60420                 return false;
60421             }
60422             return true;
60423         });
60424         
60425         return ret;
60426     },
60427     
60428     findCells : function(rec) {
60429         var s = rec.data.start_dt.clone().clearTime().getTime();
60430        // Roo.log(s);
60431         var e= rec.data.end_dt.clone().clearTime().getTime();
60432        // Roo.log(e);
60433         var ret = [];
60434         this.cells.each(function(c){
60435              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60436             
60437             if(c.dateValue > e){
60438                 return ;
60439             }
60440             if(c.dateValue < s){
60441                 return ;
60442             }
60443             ret.push(c);
60444         });
60445         
60446         return ret;    
60447     },
60448     
60449     findBestRow: function(cells)
60450     {
60451         var ret = 0;
60452         
60453         for (var i =0 ; i < cells.length;i++) {
60454             ret  = Math.max(cells[i].rows || 0,ret);
60455         }
60456         return ret;
60457         
60458     },
60459     
60460     
60461     addItem : function(rec)
60462     {
60463         // look for vertical location slot in
60464         var cells = this.findCells(rec);
60465         
60466         rec.row = this.findBestRow(cells);
60467         
60468         // work out the location.
60469         
60470         var crow = false;
60471         var rows = [];
60472         for(var i =0; i < cells.length; i++) {
60473             if (!crow) {
60474                 crow = {
60475                     start : cells[i],
60476                     end :  cells[i]
60477                 };
60478                 continue;
60479             }
60480             if (crow.start.getY() == cells[i].getY()) {
60481                 // on same row.
60482                 crow.end = cells[i];
60483                 continue;
60484             }
60485             // different row.
60486             rows.push(crow);
60487             crow = {
60488                 start: cells[i],
60489                 end : cells[i]
60490             };
60491             
60492         }
60493         
60494         rows.push(crow);
60495         rec.els = [];
60496         rec.rows = rows;
60497         rec.cells = cells;
60498         for (var i = 0; i < cells.length;i++) {
60499             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60500             
60501         }
60502         
60503         
60504     },
60505     
60506     clearEvents: function() {
60507         
60508         if (!this.eventStore.getCount()) {
60509             return;
60510         }
60511         // reset number of rows in cells.
60512         Roo.each(this.cells.elements, function(c){
60513             c.rows = 0;
60514         });
60515         
60516         this.eventStore.each(function(e) {
60517             this.clearEvent(e);
60518         },this);
60519         
60520     },
60521     
60522     clearEvent : function(ev)
60523     {
60524         if (ev.els) {
60525             Roo.each(ev.els, function(el) {
60526                 el.un('mouseenter' ,this.onEventEnter, this);
60527                 el.un('mouseleave' ,this.onEventLeave, this);
60528                 el.remove();
60529             },this);
60530             ev.els = [];
60531         }
60532     },
60533     
60534     
60535     renderEvent : function(ev,ctr) {
60536         if (!ctr) {
60537              ctr = this.view.el.select('.fc-event-container',true).first();
60538         }
60539         
60540          
60541         this.clearEvent(ev);
60542             //code
60543        
60544         
60545         
60546         ev.els = [];
60547         var cells = ev.cells;
60548         var rows = ev.rows;
60549         this.fireEvent('eventrender', this, ev);
60550         
60551         for(var i =0; i < rows.length; i++) {
60552             
60553             cls = '';
60554             if (i == 0) {
60555                 cls += ' fc-event-start';
60556             }
60557             if ((i+1) == rows.length) {
60558                 cls += ' fc-event-end';
60559             }
60560             
60561             //Roo.log(ev.data);
60562             // how many rows should it span..
60563             var cg = this.eventTmpl.append(ctr,Roo.apply({
60564                 fccls : cls
60565                 
60566             }, ev.data) , true);
60567             
60568             
60569             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60570             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60571             cg.on('click', this.onEventClick, this, ev);
60572             
60573             ev.els.push(cg);
60574             
60575             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60576             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60577             //Roo.log(cg);
60578              
60579             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60580             cg.setWidth(ebox.right - sbox.x -2);
60581         }
60582     },
60583     
60584     renderEvents: function()
60585     {   
60586         // first make sure there is enough space..
60587         
60588         if (!this.eventTmpl) {
60589             this.eventTmpl = new Roo.Template(
60590                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60591                     '<div class="fc-event-inner">' +
60592                         '<span class="fc-event-time">{time}</span>' +
60593                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60594                     '</div>' +
60595                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60596                 '</div>'
60597             );
60598                 
60599         }
60600                
60601         
60602         
60603         this.cells.each(function(c) {
60604             //Roo.log(c.select('.fc-day-content div',true).first());
60605             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60606         });
60607         
60608         var ctr = this.view.el.select('.fc-event-container',true).first();
60609         
60610         var cls;
60611         this.eventStore.each(function(ev){
60612             
60613             this.renderEvent(ev);
60614              
60615              
60616         }, this);
60617         this.view.layout();
60618         
60619     },
60620     
60621     onEventEnter: function (e, el,event,d) {
60622         this.fireEvent('evententer', this, el, event);
60623     },
60624     
60625     onEventLeave: function (e, el,event,d) {
60626         this.fireEvent('eventleave', this, el, event);
60627     },
60628     
60629     onEventClick: function (e, el,event,d) {
60630         this.fireEvent('eventclick', this, el, event);
60631     },
60632     
60633     onMonthChange: function () {
60634         this.store.load();
60635     },
60636     
60637     onLoad: function () {
60638         
60639         //Roo.log('calendar onload');
60640 //         
60641         if(this.eventStore.getCount() > 0){
60642             
60643            
60644             
60645             this.eventStore.each(function(d){
60646                 
60647                 
60648                 // FIXME..
60649                 var add =   d.data;
60650                 if (typeof(add.end_dt) == 'undefined')  {
60651                     Roo.log("Missing End time in calendar data: ");
60652                     Roo.log(d);
60653                     return;
60654                 }
60655                 if (typeof(add.start_dt) == 'undefined')  {
60656                     Roo.log("Missing Start time in calendar data: ");
60657                     Roo.log(d);
60658                     return;
60659                 }
60660                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60661                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60662                 add.id = add.id || d.id;
60663                 add.title = add.title || '??';
60664                 
60665                 this.addItem(d);
60666                 
60667              
60668             },this);
60669         }
60670         
60671         this.renderEvents();
60672     }
60673     
60674
60675 });
60676 /*
60677  grid : {
60678                 xtype: 'Grid',
60679                 xns: Roo.grid,
60680                 listeners : {
60681                     render : function ()
60682                     {
60683                         _this.grid = this;
60684                         
60685                         if (!this.view.el.hasClass('course-timesheet')) {
60686                             this.view.el.addClass('course-timesheet');
60687                         }
60688                         if (this.tsStyle) {
60689                             this.ds.load({});
60690                             return; 
60691                         }
60692                         Roo.log('width');
60693                         Roo.log(_this.grid.view.el.getWidth());
60694                         
60695                         
60696                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60697                             '.course-timesheet .x-grid-row' : {
60698                                 height: '80px'
60699                             },
60700                             '.x-grid-row td' : {
60701                                 'vertical-align' : 0
60702                             },
60703                             '.course-edit-link' : {
60704                                 'color' : 'blue',
60705                                 'text-overflow' : 'ellipsis',
60706                                 'overflow' : 'hidden',
60707                                 'white-space' : 'nowrap',
60708                                 'cursor' : 'pointer'
60709                             },
60710                             '.sub-link' : {
60711                                 'color' : 'green'
60712                             },
60713                             '.de-act-sup-link' : {
60714                                 'color' : 'purple',
60715                                 'text-decoration' : 'line-through'
60716                             },
60717                             '.de-act-link' : {
60718                                 'color' : 'red',
60719                                 'text-decoration' : 'line-through'
60720                             },
60721                             '.course-timesheet .course-highlight' : {
60722                                 'border-top-style': 'dashed !important',
60723                                 'border-bottom-bottom': 'dashed !important'
60724                             },
60725                             '.course-timesheet .course-item' : {
60726                                 'font-family'   : 'tahoma, arial, helvetica',
60727                                 'font-size'     : '11px',
60728                                 'overflow'      : 'hidden',
60729                                 'padding-left'  : '10px',
60730                                 'padding-right' : '10px',
60731                                 'padding-top' : '10px' 
60732                             }
60733                             
60734                         }, Roo.id());
60735                                 this.ds.load({});
60736                     }
60737                 },
60738                 autoWidth : true,
60739                 monitorWindowResize : false,
60740                 cellrenderer : function(v,x,r)
60741                 {
60742                     return v;
60743                 },
60744                 sm : {
60745                     xtype: 'CellSelectionModel',
60746                     xns: Roo.grid
60747                 },
60748                 dataSource : {
60749                     xtype: 'Store',
60750                     xns: Roo.data,
60751                     listeners : {
60752                         beforeload : function (_self, options)
60753                         {
60754                             options.params = options.params || {};
60755                             options.params._month = _this.monthField.getValue();
60756                             options.params.limit = 9999;
60757                             options.params['sort'] = 'when_dt';    
60758                             options.params['dir'] = 'ASC';    
60759                             this.proxy.loadResponse = this.loadResponse;
60760                             Roo.log("load?");
60761                             //this.addColumns();
60762                         },
60763                         load : function (_self, records, options)
60764                         {
60765                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60766                                 // if you click on the translation.. you can edit it...
60767                                 var el = Roo.get(this);
60768                                 var id = el.dom.getAttribute('data-id');
60769                                 var d = el.dom.getAttribute('data-date');
60770                                 var t = el.dom.getAttribute('data-time');
60771                                 //var id = this.child('span').dom.textContent;
60772                                 
60773                                 //Roo.log(this);
60774                                 Pman.Dialog.CourseCalendar.show({
60775                                     id : id,
60776                                     when_d : d,
60777                                     when_t : t,
60778                                     productitem_active : id ? 1 : 0
60779                                 }, function() {
60780                                     _this.grid.ds.load({});
60781                                 });
60782                            
60783                            });
60784                            
60785                            _this.panel.fireEvent('resize', [ '', '' ]);
60786                         }
60787                     },
60788                     loadResponse : function(o, success, response){
60789                             // this is overridden on before load..
60790                             
60791                             Roo.log("our code?");       
60792                             //Roo.log(success);
60793                             //Roo.log(response)
60794                             delete this.activeRequest;
60795                             if(!success){
60796                                 this.fireEvent("loadexception", this, o, response);
60797                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60798                                 return;
60799                             }
60800                             var result;
60801                             try {
60802                                 result = o.reader.read(response);
60803                             }catch(e){
60804                                 Roo.log("load exception?");
60805                                 this.fireEvent("loadexception", this, o, response, e);
60806                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60807                                 return;
60808                             }
60809                             Roo.log("ready...");        
60810                             // loop through result.records;
60811                             // and set this.tdate[date] = [] << array of records..
60812                             _this.tdata  = {};
60813                             Roo.each(result.records, function(r){
60814                                 //Roo.log(r.data);
60815                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60816                                     _this.tdata[r.data.when_dt.format('j')] = [];
60817                                 }
60818                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60819                             });
60820                             
60821                             //Roo.log(_this.tdata);
60822                             
60823                             result.records = [];
60824                             result.totalRecords = 6;
60825                     
60826                             // let's generate some duumy records for the rows.
60827                             //var st = _this.dateField.getValue();
60828                             
60829                             // work out monday..
60830                             //st = st.add(Date.DAY, -1 * st.format('w'));
60831                             
60832                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60833                             
60834                             var firstOfMonth = date.getFirstDayOfMonth();
60835                             var days = date.getDaysInMonth();
60836                             var d = 1;
60837                             var firstAdded = false;
60838                             for (var i = 0; i < result.totalRecords ; i++) {
60839                                 //var d= st.add(Date.DAY, i);
60840                                 var row = {};
60841                                 var added = 0;
60842                                 for(var w = 0 ; w < 7 ; w++){
60843                                     if(!firstAdded && firstOfMonth != w){
60844                                         continue;
60845                                     }
60846                                     if(d > days){
60847                                         continue;
60848                                     }
60849                                     firstAdded = true;
60850                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60851                                     row['weekday'+w] = String.format(
60852                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60853                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60854                                                     d,
60855                                                     date.format('Y-m-')+dd
60856                                                 );
60857                                     added++;
60858                                     if(typeof(_this.tdata[d]) != 'undefined'){
60859                                         Roo.each(_this.tdata[d], function(r){
60860                                             var is_sub = '';
60861                                             var deactive = '';
60862                                             var id = r.id;
60863                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60864                                             if(r.parent_id*1>0){
60865                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60866                                                 id = r.parent_id;
60867                                             }
60868                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60869                                                 deactive = 'de-act-link';
60870                                             }
60871                                             
60872                                             row['weekday'+w] += String.format(
60873                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60874                                                     id, //0
60875                                                     r.product_id_name, //1
60876                                                     r.when_dt.format('h:ia'), //2
60877                                                     is_sub, //3
60878                                                     deactive, //4
60879                                                     desc // 5
60880                                             );
60881                                         });
60882                                     }
60883                                     d++;
60884                                 }
60885                                 
60886                                 // only do this if something added..
60887                                 if(added > 0){ 
60888                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60889                                 }
60890                                 
60891                                 
60892                                 // push it twice. (second one with an hour..
60893                                 
60894                             }
60895                             //Roo.log(result);
60896                             this.fireEvent("load", this, o, o.request.arg);
60897                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60898                         },
60899                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60900                     proxy : {
60901                         xtype: 'HttpProxy',
60902                         xns: Roo.data,
60903                         method : 'GET',
60904                         url : baseURL + '/Roo/Shop_course.php'
60905                     },
60906                     reader : {
60907                         xtype: 'JsonReader',
60908                         xns: Roo.data,
60909                         id : 'id',
60910                         fields : [
60911                             {
60912                                 'name': 'id',
60913                                 'type': 'int'
60914                             },
60915                             {
60916                                 'name': 'when_dt',
60917                                 'type': 'string'
60918                             },
60919                             {
60920                                 'name': 'end_dt',
60921                                 'type': 'string'
60922                             },
60923                             {
60924                                 'name': 'parent_id',
60925                                 'type': 'int'
60926                             },
60927                             {
60928                                 'name': 'product_id',
60929                                 'type': 'int'
60930                             },
60931                             {
60932                                 'name': 'productitem_id',
60933                                 'type': 'int'
60934                             },
60935                             {
60936                                 'name': 'guid',
60937                                 'type': 'int'
60938                             }
60939                         ]
60940                     }
60941                 },
60942                 toolbar : {
60943                     xtype: 'Toolbar',
60944                     xns: Roo,
60945                     items : [
60946                         {
60947                             xtype: 'Button',
60948                             xns: Roo.Toolbar,
60949                             listeners : {
60950                                 click : function (_self, e)
60951                                 {
60952                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60953                                     sd.setMonth(sd.getMonth()-1);
60954                                     _this.monthField.setValue(sd.format('Y-m-d'));
60955                                     _this.grid.ds.load({});
60956                                 }
60957                             },
60958                             text : "Back"
60959                         },
60960                         {
60961                             xtype: 'Separator',
60962                             xns: Roo.Toolbar
60963                         },
60964                         {
60965                             xtype: 'MonthField',
60966                             xns: Roo.form,
60967                             listeners : {
60968                                 render : function (_self)
60969                                 {
60970                                     _this.monthField = _self;
60971                                    // _this.monthField.set  today
60972                                 },
60973                                 select : function (combo, date)
60974                                 {
60975                                     _this.grid.ds.load({});
60976                                 }
60977                             },
60978                             value : (function() { return new Date(); })()
60979                         },
60980                         {
60981                             xtype: 'Separator',
60982                             xns: Roo.Toolbar
60983                         },
60984                         {
60985                             xtype: 'TextItem',
60986                             xns: Roo.Toolbar,
60987                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60988                         },
60989                         {
60990                             xtype: 'Fill',
60991                             xns: Roo.Toolbar
60992                         },
60993                         {
60994                             xtype: 'Button',
60995                             xns: Roo.Toolbar,
60996                             listeners : {
60997                                 click : function (_self, e)
60998                                 {
60999                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
61000                                     sd.setMonth(sd.getMonth()+1);
61001                                     _this.monthField.setValue(sd.format('Y-m-d'));
61002                                     _this.grid.ds.load({});
61003                                 }
61004                             },
61005                             text : "Next"
61006                         }
61007                     ]
61008                 },
61009                  
61010             }
61011         };
61012         
61013         *//*
61014  * Based on:
61015  * Ext JS Library 1.1.1
61016  * Copyright(c) 2006-2007, Ext JS, LLC.
61017  *
61018  * Originally Released Under LGPL - original licence link has changed is not relivant.
61019  *
61020  * Fork - LGPL
61021  * <script type="text/javascript">
61022  */
61023  
61024 /**
61025  * @class Roo.LoadMask
61026  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61027  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61028  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61029  * element's UpdateManager load indicator and will be destroyed after the initial load.
61030  * @constructor
61031  * Create a new LoadMask
61032  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61033  * @param {Object} config The config object
61034  */
61035 Roo.LoadMask = function(el, config){
61036     this.el = Roo.get(el);
61037     Roo.apply(this, config);
61038     if(this.store){
61039         this.store.on('beforeload', this.onBeforeLoad, this);
61040         this.store.on('load', this.onLoad, this);
61041         this.store.on('loadexception', this.onLoadException, this);
61042         this.removeMask = false;
61043     }else{
61044         var um = this.el.getUpdateManager();
61045         um.showLoadIndicator = false; // disable the default indicator
61046         um.on('beforeupdate', this.onBeforeLoad, this);
61047         um.on('update', this.onLoad, this);
61048         um.on('failure', this.onLoad, this);
61049         this.removeMask = true;
61050     }
61051 };
61052
61053 Roo.LoadMask.prototype = {
61054     /**
61055      * @cfg {Boolean} removeMask
61056      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61057      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61058      */
61059     /**
61060      * @cfg {String} msg
61061      * The text to display in a centered loading message box (defaults to 'Loading...')
61062      */
61063     msg : 'Loading...',
61064     /**
61065      * @cfg {String} msgCls
61066      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61067      */
61068     msgCls : 'x-mask-loading',
61069
61070     /**
61071      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61072      * @type Boolean
61073      */
61074     disabled: false,
61075
61076     /**
61077      * Disables the mask to prevent it from being displayed
61078      */
61079     disable : function(){
61080        this.disabled = true;
61081     },
61082
61083     /**
61084      * Enables the mask so that it can be displayed
61085      */
61086     enable : function(){
61087         this.disabled = false;
61088     },
61089     
61090     onLoadException : function()
61091     {
61092         Roo.log(arguments);
61093         
61094         if (typeof(arguments[3]) != 'undefined') {
61095             Roo.MessageBox.alert("Error loading",arguments[3]);
61096         } 
61097         /*
61098         try {
61099             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61100                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61101             }   
61102         } catch(e) {
61103             
61104         }
61105         */
61106     
61107         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61108     },
61109     // private
61110     onLoad : function()
61111     {
61112         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61113     },
61114
61115     // private
61116     onBeforeLoad : function(){
61117         if(!this.disabled){
61118             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61119         }
61120     },
61121
61122     // private
61123     destroy : function(){
61124         if(this.store){
61125             this.store.un('beforeload', this.onBeforeLoad, this);
61126             this.store.un('load', this.onLoad, this);
61127             this.store.un('loadexception', this.onLoadException, this);
61128         }else{
61129             var um = this.el.getUpdateManager();
61130             um.un('beforeupdate', this.onBeforeLoad, this);
61131             um.un('update', this.onLoad, this);
61132             um.un('failure', this.onLoad, this);
61133         }
61134     }
61135 };/*
61136  * Based on:
61137  * Ext JS Library 1.1.1
61138  * Copyright(c) 2006-2007, Ext JS, LLC.
61139  *
61140  * Originally Released Under LGPL - original licence link has changed is not relivant.
61141  *
61142  * Fork - LGPL
61143  * <script type="text/javascript">
61144  */
61145
61146
61147 /**
61148  * @class Roo.XTemplate
61149  * @extends Roo.Template
61150  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61151 <pre><code>
61152 var t = new Roo.XTemplate(
61153         '&lt;select name="{name}"&gt;',
61154                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61155         '&lt;/select&gt;'
61156 );
61157  
61158 // then append, applying the master template values
61159  </code></pre>
61160  *
61161  * Supported features:
61162  *
61163  *  Tags:
61164
61165 <pre><code>
61166       {a_variable} - output encoded.
61167       {a_variable.format:("Y-m-d")} - call a method on the variable
61168       {a_variable:raw} - unencoded output
61169       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61170       {a_variable:this.method_on_template(...)} - call a method on the template object.
61171  
61172 </code></pre>
61173  *  The tpl tag:
61174 <pre><code>
61175         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61176         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61177         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61178         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61179   
61180         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61181         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61182 </code></pre>
61183  *      
61184  */
61185 Roo.XTemplate = function()
61186 {
61187     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61188     if (this.html) {
61189         this.compile();
61190     }
61191 };
61192
61193
61194 Roo.extend(Roo.XTemplate, Roo.Template, {
61195
61196     /**
61197      * The various sub templates
61198      */
61199     tpls : false,
61200     /**
61201      *
61202      * basic tag replacing syntax
61203      * WORD:WORD()
61204      *
61205      * // you can fake an object call by doing this
61206      *  x.t:(test,tesT) 
61207      * 
61208      */
61209     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61210
61211     /**
61212      * compile the template
61213      *
61214      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61215      *
61216      */
61217     compile: function()
61218     {
61219         var s = this.html;
61220      
61221         s = ['<tpl>', s, '</tpl>'].join('');
61222     
61223         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61224             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61225             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61226             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61227             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61228             m,
61229             id     = 0,
61230             tpls   = [];
61231     
61232         while(true == !!(m = s.match(re))){
61233             var forMatch   = m[0].match(nameRe),
61234                 ifMatch   = m[0].match(ifRe),
61235                 execMatch   = m[0].match(execRe),
61236                 namedMatch   = m[0].match(namedRe),
61237                 
61238                 exp  = null, 
61239                 fn   = null,
61240                 exec = null,
61241                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61242                 
61243             if (ifMatch) {
61244                 // if - puts fn into test..
61245                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61246                 if(exp){
61247                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61248                 }
61249             }
61250             
61251             if (execMatch) {
61252                 // exec - calls a function... returns empty if true is  returned.
61253                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61254                 if(exp){
61255                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61256                 }
61257             }
61258             
61259             
61260             if (name) {
61261                 // for = 
61262                 switch(name){
61263                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61264                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61265                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61266                 }
61267             }
61268             var uid = namedMatch ? namedMatch[1] : id;
61269             
61270             
61271             tpls.push({
61272                 id:     namedMatch ? namedMatch[1] : id,
61273                 target: name,
61274                 exec:   exec,
61275                 test:   fn,
61276                 body:   m[1] || ''
61277             });
61278             if (namedMatch) {
61279                 s = s.replace(m[0], '');
61280             } else { 
61281                 s = s.replace(m[0], '{xtpl'+ id + '}');
61282             }
61283             ++id;
61284         }
61285         this.tpls = [];
61286         for(var i = tpls.length-1; i >= 0; --i){
61287             this.compileTpl(tpls[i]);
61288             this.tpls[tpls[i].id] = tpls[i];
61289         }
61290         this.master = tpls[tpls.length-1];
61291         return this;
61292     },
61293     /**
61294      * same as applyTemplate, except it's done to one of the subTemplates
61295      * when using named templates, you can do:
61296      *
61297      * var str = pl.applySubTemplate('your-name', values);
61298      *
61299      * 
61300      * @param {Number} id of the template
61301      * @param {Object} values to apply to template
61302      * @param {Object} parent (normaly the instance of this object)
61303      */
61304     applySubTemplate : function(id, values, parent)
61305     {
61306         
61307         
61308         var t = this.tpls[id];
61309         
61310         
61311         try { 
61312             if(t.test && !t.test.call(this, values, parent)){
61313                 return '';
61314             }
61315         } catch(e) {
61316             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61317             Roo.log(e.toString());
61318             Roo.log(t.test);
61319             return ''
61320         }
61321         try { 
61322             
61323             if(t.exec && t.exec.call(this, values, parent)){
61324                 return '';
61325             }
61326         } catch(e) {
61327             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61328             Roo.log(e.toString());
61329             Roo.log(t.exec);
61330             return ''
61331         }
61332         try {
61333             var vs = t.target ? t.target.call(this, values, parent) : values;
61334             parent = t.target ? values : parent;
61335             if(t.target && vs instanceof Array){
61336                 var buf = [];
61337                 for(var i = 0, len = vs.length; i < len; i++){
61338                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61339                 }
61340                 return buf.join('');
61341             }
61342             return t.compiled.call(this, vs, parent);
61343         } catch (e) {
61344             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61345             Roo.log(e.toString());
61346             Roo.log(t.compiled);
61347             return '';
61348         }
61349     },
61350
61351     compileTpl : function(tpl)
61352     {
61353         var fm = Roo.util.Format;
61354         var useF = this.disableFormats !== true;
61355         var sep = Roo.isGecko ? "+" : ",";
61356         var undef = function(str) {
61357             Roo.log("Property not found :"  + str);
61358             return '';
61359         };
61360         
61361         var fn = function(m, name, format, args)
61362         {
61363             //Roo.log(arguments);
61364             args = args ? args.replace(/\\'/g,"'") : args;
61365             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61366             if (typeof(format) == 'undefined') {
61367                 format= 'htmlEncode';
61368             }
61369             if (format == 'raw' ) {
61370                 format = false;
61371             }
61372             
61373             if(name.substr(0, 4) == 'xtpl'){
61374                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61375             }
61376             
61377             // build an array of options to determine if value is undefined..
61378             
61379             // basically get 'xxxx.yyyy' then do
61380             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61381             //    (function () { Roo.log("Property not found"); return ''; })() :
61382             //    ......
61383             
61384             var udef_ar = [];
61385             var lookfor = '';
61386             Roo.each(name.split('.'), function(st) {
61387                 lookfor += (lookfor.length ? '.': '') + st;
61388                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61389             });
61390             
61391             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61392             
61393             
61394             if(format && useF){
61395                 
61396                 args = args ? ',' + args : "";
61397                  
61398                 if(format.substr(0, 5) != "this."){
61399                     format = "fm." + format + '(';
61400                 }else{
61401                     format = 'this.call("'+ format.substr(5) + '", ';
61402                     args = ", values";
61403                 }
61404                 
61405                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61406             }
61407              
61408             if (args.length) {
61409                 // called with xxyx.yuu:(test,test)
61410                 // change to ()
61411                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61412             }
61413             // raw.. - :raw modifier..
61414             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61415             
61416         };
61417         var body;
61418         // branched to use + in gecko and [].join() in others
61419         if(Roo.isGecko){
61420             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61421                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61422                     "';};};";
61423         }else{
61424             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61425             body.push(tpl.body.replace(/(\r\n|\n)/g,
61426                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61427             body.push("'].join('');};};");
61428             body = body.join('');
61429         }
61430         
61431         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61432        
61433         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61434         eval(body);
61435         
61436         return this;
61437     },
61438
61439     applyTemplate : function(values){
61440         return this.master.compiled.call(this, values, {});
61441         //var s = this.subs;
61442     },
61443
61444     apply : function(){
61445         return this.applyTemplate.apply(this, arguments);
61446     }
61447
61448  });
61449
61450 Roo.XTemplate.from = function(el){
61451     el = Roo.getDom(el);
61452     return new Roo.XTemplate(el.value || el.innerHTML);
61453 };